PHP生成符合DTD的XML需严格遵循DTD定义的结构、顺序、属性和内容模型,使用DOMDocument精确构建元素与属性,并通过外部工具或DOMDocument::validate()校验。

PHP 生成符合特定 DTD 的 XML,核心在于手动或半自动地确保结构、元素顺序、属性声明、内容模型与 DTD 完全一致。PHP 本身不校验 DTD,也不自动生成 DTD 兼容结构,需开发者主动遵循规则编写或借助工具辅助验证。
明确 DTD 约束细节再编码
DTD 定义了哪些元素可出现、是否必需、能否嵌套、属性类型及默认值等。例如:
这意味着:
• book 必须包含且仅包含一个 title、至少一个 author、一个 price(顺序不可变)
• book 元素必须带 id 属性,且值全局唯一
• title、author、price 只能含纯文本(无子元素)
生成 XML 前,先对照 DTD 梳理数据结构和输出逻辑,避免遗漏或错序。
用 DOMDocument 构建结构化 XML
DOM 扩展能精确控制元素层级、属性和顺序,比字符串拼接更可靠:
立即学习“PHP免费学习笔记(深入)”;
- 创建
DOMDocument实例,设置xmlVersion和encoding - 用
createElement()和appendChild()严格按 DTD 规定顺序添加子节点 - 用
setAttribute()添加必需属性(如ID类型属性要确保值合法:字母/下划线开头,仅含字母数字或连字符) - 对
#PCDATA内容,用createTextNode()插入,避免直接赋值导致转义错误
示例关键片段:
$doc = new DOMDocument('1.0', 'UTF-8');
$book = $doc->createElement('book');
$book->setAttribute('id', 'bk001'); // 符合 ID 约束
$title = $doc->createElement('title');
$title->appendChild($doc->createTextNode('PHP Guide'));
$book->appendChild($title);
$author = $doc->createElement('author');
$author->appendChild($doc->createTextNode('Alice'));
$book->appendChild($author); // 至少一个,可循环追加
$price = $doc->createElement('price');
$price->appendChild($doc->createTextNode('29.99'));
$book->appendChild($price);
$doc->appendChild($book);
输出前关联 DTD 并验证(可选但推荐)
生成 XML 字符串后,若需强校验,可:
- 在
saveXML()前调用documentElement->setAttribute('DOCTYPE', ...)—— 实际不生效;正确方式是手动在 XML 头部插入 DTD 声明(注意:DOM 不自动维护外部 DTD 引用) - 将生成的 XML 保存为文件,用命令行工具(如
xmllint --dtdvalid my.dtd document.xml)或 PHP 的libxml_use_internal_errors(true)+DOMDocument::validate()进行离线校验 -
DOMDocument::validate()要求 DTD 已加载(如通过loadHTMLFile()或解析含内联 DTD 的文档),对独立 XML+外部 DTD 需配合libxml_set_external_entity_loader控制实体解析
避免常见陷阱
• ID 属性值非法:不能以数字开头,不能含空格或特殊符号,否则 DTD 校验失败
• 顺序错乱:DTD 中 (A, B+, C) 表示 B 必须在 A 后、C 前,DOM 中 append 顺序必须匹配
• 空元素误写:DTD 若定义 price (#PCDATA),就不能输出 ,而应为 或
• 未转义特殊字符:用 createTextNode() 自动处理 、& 等;若手动拼接,务必用 htmlspecialchars()(但注意不要双重转义)











