PHP如何生成符合特定DTD的XML

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

PHP 生成符合特定 DTD 的 XML,核心在于手动或半自动地确保结构、元素顺序、属性声明、内容模型与 DTD 完全一致。PHP 本身不校验 DTD,也不自动生成 DTD 兼容结构,需开发者主动遵循规则编写或借助工具辅助验证。

明确 DTD 约束细节再编码

DTD 定义了哪些元素可出现、是否必需、能否嵌套、属性类型及默认值等。例如:





这意味着:
book 必须包含且仅包含一个 title、至少一个 author、一个 price(顺序不可变)
book 元素必须带 id 属性,且值全局唯一
titleauthorprice 只能含纯文本(无子元素)

生成 XML 前,先对照 DTD 梳理数据结构和输出逻辑,避免遗漏或错序。

用 DOMDocument 构建结构化 XML

DOM 扩展能精确控制元素层级、属性和顺序,比字符串拼接更可靠:

  • 创建 DOMDocument 实例,设置 xmlVersionencoding
  • 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),就不能输出 ,而应为 0.00
未转义特殊字符:用 createTextNode() 自动处理 & 等;若手动拼接,务必用 htmlspecialchars()(但注意不要双重转义)