记住 DOMNode 始终需要对父节点或 DOMDocument 的引用。
例如,如果您尝试克隆一个节点 - 复制所有子节点 - 并覆盖或删除保存克隆节点的变量 - 所有子节点将丢失任何引用并变得无效。
如果您很幸运,这将导致一条类似“无法... 节点不再存在”的提示消息。在大多数情况下,PHP 仅提供给您“无法获取 DOM[...]”的简略信息,这使得难以找出发生了什么,具体取决于当前的操作。
(PHP 5, PHP 7, PHP 8)
DOMNode::cloneNode — 克隆节点
克隆的节点。
记住 DOMNode 始终需要对父节点或 DOMDocument 的引用。
例如,如果您尝试克隆一个节点 - 复制所有子节点 - 并覆盖或删除保存克隆节点的变量 - 所有子节点将丢失任何引用并变得无效。
如果您很幸运,这将导致一条类似“无法... 节点不再存在”的提示消息。在大多数情况下,PHP 仅提供给您“无法获取 DOM[...]”的简略信息,这使得难以找出发生了什么,具体取决于当前的操作。
<?php
//@oliver 感谢示例源代码...
/*
cloneNode(false) 不会省略
克隆节点的属性,
要实现这一点,需要进行迭代。
这可能效率低于
仅仅从所需的 nodeName
创建新节点
但在某些情况下可能很有用。
用例:
省略 xml 文档
安全部分的子节点和属性
而不改变预期的整体结构;
*/
//要使用的 xml
$file="<?xml version='1.0'?>
<book type='paperback'>
<title name='MAP'>Red Nails</title>
<price>$12.99</price>
<author>
<name first='Robert' middle='E' last='Howard'/>
<birthdate disco='false' nirvana='definitely'>
9/21/1977
<month title='september' />
</birthdate>
</author>
<author>
<name first='Arthur' middle='Mc' last='Kayn'/>
</author>
</book>";
$doc = new domDocument;
$doc->loadXML($file);
$xpath = new domXPath($doc);
$query = "//author/birthdate";
$xpathQuery = $xpath->query($query);
//在生产代码中将是一个循环...
$child = $xpathQuery->item(0);
$parent = $child->parentNode;
$doppel = $child->cloneNode(false);
$limit = $doppel->attributes->length;
for ($a=0;$a<$limit;$a++) {
$doppel->removeAttributeNode($doppel->attributes->item(0));
}
//现在交换为空节点
$parent->replaceChild( $doppel, $child);
print $doc->saveXML();
?>
如果您需要克隆包含所有子 DOMNode 元素的节点
private function cloneNode($node){
$nd = new DOMNode();
for ($i = 0; $i < $node->childNodes->length; $i++) {
$child = $node->childNodes->item($i);
if ($child->nodeType === XML_TEXT_NODE) {
$nd->appendChild($node->cloneNode(true));
}
else{
$nd->appendChild($this->cloneNode($child));
}
}
return $nd;
}
如果您需要一些函数来克隆节点而不触及命名空间,您可以使用以下方法。
<?php
私有函数 cloneNode($node,$doc){
$nd=$doc->createElement($node->nodeName);
遍历($node->attributes 作为 $value)
$nd->setAttribute($value->nodeName,$value->value);
如果(!$node->childNodes)
返回 $nd;
遍历($node->childNodes 作为 $child) {
如果($child->nodeName=="#text")
$nd->appendChild($doc->createTextNode($child->nodeValue));
否则
$nd->appendChild(cloneNode($child,$doc));
}
返回 $nd;
}
?>
如果您有一个包含DOMNode的对象,克隆该对象不会将DOMNode与其一起克隆。如果您只是复制该对象或多次添加其DOMNode,实际上您只会移动树中的DOMNode,而不是复制它。这似乎很明显,但我花了半天时间才发现。
该对象需要使用__clone并手动克隆节点
<?php
类 containsNode {
公共 $node; //从某个地方设置
公共函数 __clone() {
$this->node = $this->node->cloneNode(TRUE);
}
}
?>
节点克隆的简单示例
<?xml version="1.0"?>
<book type="paperback">
<title name='MAP'>Red Nails</title>
<price>$12.99</price>
<author>
<name first="Robert" middle="E" last="Howard"/>
<birthdate>9/21/1977</birthdate>
</author>
<author>
<name first="Arthur" middle="Mc" last="Kayn"/>
</author>
</book>
<?php
//要使用的xml文件名
$file = 'book.xml';
$doc = new domDocument;
如果 (file_exists($file)) {
$doc->load($file);
} 否则 {
退出('错误!.');
}
$xpath = new domXPath($doc);
$query = "//author/*";
$xpathQuery = $xpath->query($query);
$size = $xpathQuery->length;
对于 ($i=0; $i<$size; $i++){
$node = $xpathQuery->item($i);
如果 ($node->nodeName == 'birthdate' && $node->hasChildNodes() && $node->firstChild->textContent != ''){
$clonenode = $node->cloneNode(true);
$refnode = $node;
}
}
对于 ($i=0; $i<$size; $i++){
$node = $xpathQuery->item($i);
如果 (!$node->isSameNode($refnode)){
$newnode = $node->appendChild($clonenode);
}
}
打印 $doc->saveXML();
?>