PHP Conference Japan 2024

DOMNode::cloneNode

(PHP 5, PHP 7, PHP 8)

DOMNode::cloneNode 克隆节点

描述

public DOMNode::cloneNode(bool $deep = false): DOMNode|false

创建节点的副本。

参数

deep

指示是否复制所有后代节点。此参数默认为 false

返回值

克隆的节点。

添加注释

用户贡献的注释 6 条注释

frame at dynamiccreated dot de
12 年前
记住 DOMNode 始终需要对父节点或 DOMDocument 的引用。

例如,如果您尝试克隆一个节点 - 复制所有子节点 - 并覆盖或删除保存克隆节点的变量 - 所有子节点将丢失任何引用并变得无效。

如果您很幸运,这将导致一条类似“无法... 节点不再存在”的提示消息。在大多数情况下,PHP 仅提供给您“无法获取 DOM[...]”的简略信息,这使得难以找出发生了什么,具体取决于当前的操作。
[montana] at [percepticon] dot [com]
15 年前
<?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();

?>
cmd at 1xinternet dot de
8 年前
如果您需要克隆包含所有子 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;
}
cemkalyoncu at gmail dot com
15 年前
如果您需要一些函数来克隆节点而不触及命名空间,您可以使用以下方法。

<?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;
}
?>
[email protected]
10年前
如果您有一个包含DOMNode的对象,克隆该对象不会将DOMNode与其一起克隆。如果您只是复制该对象或多次添加其DOMNode,实际上您只会移动树中的DOMNode,而不是复制它。这似乎很明显,但我花了半天时间才发现。

该对象需要使用__clone并手动克隆节点

<?php
containsNode {
公共
$node; //从某个地方设置

公共函数 __clone() {
$this->node = $this->node->cloneNode(TRUE);
}
}
?>
[email protected]
20年前
节点克隆的简单示例

<?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();

?>
To Top