为什么 xml.dom.minidom walk to remove empty text nodes 工作意外?

Why does xml.dom.minidom walk to remove empty text nodes work unexpectedly?

代码:(通常深度优先)

import xml.dom.minidom as xdom    

def _walk_n_apply(func, cond, parent):                                                                                                                                                          
    if parent.childNodes:                                                                                                                                                                       
       for child in parent.childNodes:                                                                                                                                                          
            if cond(child):                                                                                                                                                                    
                func(parent, child)                                                                                                                                                              
                continue                                                                                                                                                                      
            _walk_n_apply(func, cond, child)                                                                                                                                                   

def remove_child(parent, child):                                                                                                                                                                 
    node = parent.removeChild(child)                                                                                                                                                             
    print 'removed', node                                                                                                                                                                     

def is_empty_text_node(node):                                                                                                                                                                 
    return node.nodeType == node.TEXT_NODE and node.data.strip() == ''   


xmldom = xdom.parse('blah')

_walk_n_apply(remove_child, is_empty_text_node, xmldom)

在Ipython中,调用

_walk_n_apply(remove_child, is_empty_text_node, xmldom)

一次,输出有轻微变化:

print xmldom.toprettyxml()

但是,如果我多次调用它,"several depending on the level of nesting",它最终会给出一个格式很好的 prettyxml

如何通过一次调用实现此目的?


输入文件内容:

<grammar xmlns="http://www.w3.org/2001/06/grammar"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://www.w3.org/2001/06/grammar
                             http://www.w3.org/TR/speech-grammar/grammar.xsd"
         xml:lang="en" version="1.0"
         root="command"
         mode="voice"
         tag-format="semantics/1.0">

<rule id="command">
   <one-of>
      <item><ruleref uri="#announcement" /></item>
      <item><ruleref uri="#hello" /></item>
      <item><ruleref uri="#whereis" /></item>
      <item><ruleref uri="#interrupt" /></item>
      <item><ruleref uri="#message" /></item>
      <item><ruleref uri="#logon" /></item>
      <item><ruleref uri="#logoff" /></item>
      <item><ruleref uri="#storecoverage" /></item>
      <item><ruleref uri="#identify" /></item>
      <item><ruleref uri="#near" /></item>
      <item><ruleref uri="#time" /></item>
      <item><ruleref uri="#playmessages" /></item>
      <item><ruleref uri="#registerbackup" /></item>
      <item><ruleref uri="#igotit" /></item>
   </one-of>
   <tag>out=rules.latest()</tag>
</rule>

<rule id="announcement">
<item>
  <one-of>
   <item>announcement today<tag>out="AnnouncementToday"</tag></item>
   <item>announcement now<tag>out="AnnouncementNow"</tag></item>
   <item>announcement hour<tag>out="AnnouncementHour"</tag></item>
  </one-of>
</item>
</rule>

<rule id="hello">
  <item repeat="0-1">
    <one-of>
        <item>hello</item>
        <item>hey</item>
        <item>hi</item>
    </one-of>
   </item>
   <item><ruleref uri="persons.grxml"/><tag>out="Hello,"+rules.latest()</tag></item>
</rule>

<rule id="whereis">
  <item>
   <one-of>
      <item>where is<ruleref uri="persons.grxml"/></item>
      <item>locate<ruleref uri="persons.grxml"/></item>
      <item>find<ruleref uri="persons.grxml"/></item>
   </one-of>
   <tag>out="Whereis,"+rules.latest()</tag>
  </item>
</rule>

<rule id="interrupt">
   <item>interrupt<ruleref uri="persons.grxml"/><tag>out="Interrupt,"+rules.latest()</tag></item>
</rule>

<rule id="message">
<item>message</item>
  <item repeat="0-1">for</item>
   <item><ruleref uri="persons.grxml"/><tag>out="Message,"+rules.latest()</tag></item>
</rule>

<rule id="logon">
   <one-of>
    <item>log on
       <one-of>
        <item><ruleref uri="persons.grxml"/><tag>out="Logon,"+rules.latest()</tag></item>
        <item><ruleref uri="#id_numbers"/><tag>out="Logon,"+rules.latest()</tag></item>
       </one-of>
        </item>
   </one-of>
</rule>

<rule id="logoff">
<item>
  <one-of>
      <item>log off<item repeat='0-1'>system</item></item>
      <item>log out</item>
  </one-of>
  <tag>out="Logoff"</tag>
</item>
</rule>

<rule id="storecoverage">
   <item repeat="0-1">store</item>
    <item>coverage<tag>out="coverage"</tag></item>
</rule>

<rule id="identify">
   <item>identify<tag>out="identify"</tag></item>
</rule>

<rule id="near">
    <one-of>
      <item>who is</item>
      <item>anyone</item>
    </one-of>
  <item>near<ruleref uri="#locations"/><tag>out="near,"+rules.latest()</tag></item>
</rule>

<rule id="time">
<one-of>
    <item>time<tag>out="time"</tag></item>
    <item>what time is it<tag>out="time"</tag></item>
</one-of>
</rule>

<rule id="playmessages">
  <item>
    play
   <one-of>
      <item>messages<tag>out="PlayMessages"</tag></item>
      <item>announcements<tag>out="PlayMessages"</tag></item>
   </one-of>
  </item>
</rule>

<rule id="registerbackup">
   <item repeat="0-1">cash</item>
    <item>register backup<tag>out="register backup"</tag></item>
</rule>

<rule id="igotit">
 <one-of>
   <item>
    <one-of>
     <item>i got it<tag>out="i got it"</tag></item>
     <item>i have it<tag>out="i got it"</tag></item>
    </one-of>
   </item>
   <item>
    <one-of>
     <item>on the way<tag>out="i got it"</tag></item>
     <item>on my way<tag>out="i got it"</tag></item>
    </one-of>
   </item> 
 </one-of>
</rule>


<rule id="locations">
   <ruleref uri="locations.grxml"/>
   <tag>out=rules.latest();</tag>
</rule>

如果我只调用函数一次输出:

removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">

如果我迭代调用该函数 10 次,输出:

像这样

for i in range(10):
    _walk_n_apply(remove_child, is_empty_text_node, xmldom)

(输出是从 tmux 会话中复制粘贴的,因此可能会遗漏几行;我不理解的是,如果我的函数是递归且正确的,它应该已经消除了所有空的-一次调用中的文本节点。但是第二次调用它也会导致删除更多的空文本节点,然后第三次,依此类推...直到没有更多的空文本节点为止。)

removed <DOM Text node "u'\n\n'">                                                                                                                                                       
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n  '">                                                                                                                                                       
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n    '">                                                                                                                                                     
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u' \n '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">                                                                                                                                                    
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n'">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n       '">
removed <DOM Text node "u'\n        '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n      '">
removed <DOM Text node "u'\n  '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n   '">
removed <DOM Text node "u'\n     '">
removed <DOM Text node "u'\n     '">
removed <DOM Text node "u'\n    '">
removed <DOM Text node "u'\n     '">
removed <DOM Text node "u'\n     '">
removed <DOM Text node "u'\n    '">

您在迭代 .childNodes 时正在操纵子项列表。试试这个:

for child in list(parent.childNodes):