最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

xpath - How do I iterate through a javascript XPathResult that is of type ORDERED_NODE_SNAPSHOT_TYPE - Stack Overflow

programmeradmin1浏览0评论

I need to get an XPathResult with javascript and iterate through it, cloning each node of the result. Initially, I tried the following with the result as ORDERED_NODE_ITERATOR_TYPE:

childNodesXPath = '//div[@id="'+subcat_id+'" and @parentid="'+subcat_parent_id+'"]';    
subcat_child_nodes = document.evaluate(childNodesXPath, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);
while (next_child_node = subcat_child_nodes.iterateNext()) {
    new_child_node = next_child_node.cloneNode(true);
    new_child_node.setAttribute('parentid', target_id);
    new_child_node.setAttribute('grandparentid', target_parentid);
    new_length = new_subcat_child_nodes.push(new_child_node);
}

Of course I discovered that the iterator became invalid as soon as the first node was cloned because the DOM changed, so then I tried this with the result as ORDERED_NODE_SNAPSHOT_TYPE:

childNodesXPath = '//div[@id="'+subcat_id+'" and @parentid="'+subcat_parent_id+'"]';
subcat_child_nodes = document.evaluate(childNodesXPath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
for (i=0; i<subcat_child_nodes.length; i++) {
    new_child_node = subcat_child_nodes[i].cloneNode(true);
    new_child_node.setAttribute('parentid', target_id);
    new_child_node.setAttribute('grandparentid', target_parentid);
    new_length = new_subcat_child_nodes.push(new_child_node);
}

This did not work because there is no length property for the XPathResult object. I also tried subcat_child_nodes.forEach() and that did not work, nor does iterateNext().

How do I iterate through an XPathResult that is of type ORDERED_NODE_SNAPSHOT_TYPE in a way that allows me to clone each node? If that is not possible, is there a way to clone an entire XPathResult that is a list of nodes?

I need to get an XPathResult with javascript and iterate through it, cloning each node of the result. Initially, I tried the following with the result as ORDERED_NODE_ITERATOR_TYPE:

childNodesXPath = '//div[@id="'+subcat_id+'" and @parentid="'+subcat_parent_id+'"]';    
subcat_child_nodes = document.evaluate(childNodesXPath, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);
while (next_child_node = subcat_child_nodes.iterateNext()) {
    new_child_node = next_child_node.cloneNode(true);
    new_child_node.setAttribute('parentid', target_id);
    new_child_node.setAttribute('grandparentid', target_parentid);
    new_length = new_subcat_child_nodes.push(new_child_node);
}

Of course I discovered that the iterator became invalid as soon as the first node was cloned because the DOM changed, so then I tried this with the result as ORDERED_NODE_SNAPSHOT_TYPE:

childNodesXPath = '//div[@id="'+subcat_id+'" and @parentid="'+subcat_parent_id+'"]';
subcat_child_nodes = document.evaluate(childNodesXPath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
for (i=0; i<subcat_child_nodes.length; i++) {
    new_child_node = subcat_child_nodes[i].cloneNode(true);
    new_child_node.setAttribute('parentid', target_id);
    new_child_node.setAttribute('grandparentid', target_parentid);
    new_length = new_subcat_child_nodes.push(new_child_node);
}

This did not work because there is no length property for the XPathResult object. I also tried subcat_child_nodes.forEach() and that did not work, nor does iterateNext().

How do I iterate through an XPathResult that is of type ORDERED_NODE_SNAPSHOT_TYPE in a way that allows me to clone each node? If that is not possible, is there a way to clone an entire XPathResult that is a list of nodes?

Share Improve this question edited Apr 19, 2018 at 22:42 kalinma asked Apr 19, 2018 at 22:36 kalinmakalinma 5276 silver badges17 bronze badges 4
  • 3 Everything you wanted to know about XPath but were too afraid to search for ... hint, .snapshotLength and .snapshotItem() - which you can use to create an array if you like: const array = Array.from({length: subcat_child_nodes .snapshotLength}, (_, i) => subcat_child_nodes.snapshotItem(i)); – Jaromanda X Commented Apr 19, 2018 at 22:47
  • 3 Actually, this link is probably a better intro to xpath – Jaromanda X Commented Apr 19, 2018 at 22:54
  • Thanks, Jaromanda. That was very helpful. – kalinma Commented Apr 20, 2018 at 18:25
  • If you'd like to copy your ment to an answer, I can give you credit for the correct answer. – kalinma Commented Apr 20, 2018 at 18:32
Add a ment  | 

2 Answers 2

Reset to default 8

So, just in case anyone else is searching for the answer to my question above, Jaromanda's answer in the ments pointed me to a reference resource (archive) and this is what I ended up using.

subcat_child_nodes = document.evaluate(childNodesXPath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
for (i=0; i<subcat_child_nodes.snapshotLength; i++) {
    new_child_node = subcat_child_nodes.snapshotItem(i).cloneNode(true);
    new_child_node.setAttribute('parentid', target_id);
    new_child_node.setAttribute('grandparentid', target_parentid);
    new_length = new_subcat_child_nodes.push(new_child_node);
}

late example for newers if needed:

function iterateXPathResult() {
  let xpath = " //span[text()='Ad'] //ancestor::article";
  let nodes = document.evaluate(xpath, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);

  try {
    let node = nodes.iterateNext();
    while (node) {
      console.log("node", node);
      node = nodes.iterateNext();
    }
  } catch (e) {
    console.error(`Document tree modified during iteration: ${e}`);
  }
}

function iterateXPathResult() {
  let xpath = " //span[text()='Ad'] //ancestor::article";
  let nodes = document.evaluate(xpath, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);

  try {
    let node = nodes.iterateNext();
    while (node) {
      console.log("node", node);
      node = nodes.iterateNext();
    }
  } catch (e) {
    console.error(`Document tree modified during iteration: ${e}`);
    console.warn(`If you don't want to see this error close console on Code Snippet web and open console in your Browser`);
  }
}

iterateXPathResult();
<div class='main'>
  <article id='id-article-1'>
    <span>article 1</span>
  </article>
  <article id='id-article-2'>
    <span>Ad</span>
    <span> 1</span>
  </article>
  <article id='id-article-3'>
    <span>article 3</span>
  </article>
  <article id='id-article-4'>
    <span>Ad</span>
    <span> 2</span>
  </article>
  <article id='id-article-5'>
    <span>article 5</span>
  </article>
  Open the browser console to see the results!
</div>

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论