访问 ElementTree 节点的父节点

2025-02-20 09:23:00
admin
原创
26
摘要:问题描述:我正在使用内置的 Python ElementTree 模块。访问子节点很简单,但是父节点或兄弟节点呢? - 可以在不遍历整个树的情况下有效地完成此操作吗?解决方案 1:没有以属性形式提供的直接支持,但您也许可以使用此处parent描述的模式来实现所需的效果。建议使用以下一行代码(从链接的帖子更新到...

问题描述:

我正在使用内置的 Python ElementTree 模块。访问子节点很简单,但是父节点或兄弟节点呢? - 可以在不遍历整个树的情况下有效地完成此操作吗?


解决方案 1:

没有以属性形式提供的直接支持,但您也许可以使用此处parent描述的模式来实现所需的效果。建议使用以下一行代码(从链接的帖子更新到 Python 3.8)为整个树创建子到父的映射,使用方法:xml.etree.ElementTree.Element.iter

parent_map = {c: p for p in tree.iter() for c in p}

解决方案 2:

Vinay 的答案应该仍然有效,但对于 Python 2.7+ 和 3.2+,建议如下:

parent_map = {c:p for p in tree.iter() for c in p}

getiterator()已被弃用,取而代之的是iter(),最好使用新的dict列表理解构造函数。

其次,在构建 XML 文档时,一个子文档可能会有多个父文档,不过一旦序列化文档,父文档就会被删除。如果这很重要,您可以尝试以下方法:

parent_map = {}
for p in tree.iter():
    for c in p:
        if c in parent_map:
            parent_map[c].append(p)
            # Or raise, if you don't want to allow this.
        else:
            parent_map[c] = [p]
            # Or parent_map[c] = p if you don't want to allow this

解决方案 3:

您可以...在 ElementTree 中使用 xpath 符号。

<parent>
     <child id="123">data1</child>
</parent>

xml.findall('.//child[@id="123"]...')
>> [<Element 'parent'>]

解决方案 4:

正如在使用 find 方法 (xml.etree.ElementTree) 后获取父元素中提到的那样,您必须对父元素进行间接搜索。拥有 xml:

<a>
 <b>
  <c>data</c>
  <d>data</d>    
 </b>
</a>

假设您已经将 etree 元素创建到xml变量中,则可以使用:

 In[1] parent = xml.find('.//c/..')
 In[2] child = parent.find('./c')

导致:

Out[1]: <Element 'b' at 0x00XXXXXX> 
Out[2]: <Element 'c' at 0x00XXXXXX>

较高等级的父母将被发现为secondparent=xml.find('.//c/../..')<Element 'a' at 0x00XXXXXX>

解决方案 5:

在这里粘贴我的回答来自https://stackoverflow.com/a/54943960/492336

我遇到过类似的问题,于是想出了一些办法。结果发现没有什么可以阻止我们自己添加父级信息。我们可以在不再需要时将其删除。

def addParentInfo(et):
    for child in et:
        child.attrib['__my_parent__'] = et
        addParentInfo(child)

def stripParentInfo(et):
    for child in et:
        child.attrib.pop('__my_parent__', 'None')
        stripParentInfo(child)

def getParent(et):
    if '__my_parent__' in et.attrib:
        return et.attrib['__my_parent__']
    else:
        return None

# Example usage

tree = ...
addParentInfo(tree.getroot())
el = tree.findall(...)[0]
parent = getParent(el)
while parent:
    doSomethingWith(parent)
    parent = getParent(parent)
stripParentInfo(tree.getroot())

解决方案 6:

XPath'..' 选择器不能用于在 3.5.3 和 3.6.1 上检索父节点(至少在 OSX 上),例如在交互模式下:

import xml.etree.ElementTree as ET
root = ET.fromstring('<parent><child></child></parent>')
child = root.find('child')
parent = child.find('..') # retrieve the parent
parent is None # unexpected answer True

最后的答案打破了所有希望......

解决方案 7:

得到了以下人的答复

https://towardsdatascience.com/processing-xml-in-python-elementtree-c8992941efd2

提示:在 XPath 中使用‘...’返回当前元素的父元素。


for object_book in root.findall('.//*[@name="The Hunger Games"]...'):
    print(object_book)

解决方案 8:

迄今为止发布的大多数解决方案

  • 要么使用 XPath…但 Python 一般不支持使用 XPath 查找祖先(见注释),

  • 或者在构建整个树之后对其进行后处理(例如这个答案或那个答案)...但这需要解析和构建整个树,这对于大型 XML 数据(例如 Wikipedia 转储)来说可能是不可取的。

如果您以增量方式解析 XML,例如使用xml.etree.ElementTree.iterparse
或,则可以通过跟踪标签(和事件)xml.etree.ElementTree.XMLPullParser的开始和结束来跟踪当前路径(从根节点到当前节点)。例如:start`end`

import xml.etree.ElementTree as ET

current_path = [ ]

for event, elem in ET.iterparse('test.xml', events=['start', 'end']):
    # opening tag:
    if event == 'start':
        current_path.append(elem)
    # closing tag:
    else:
        assert event == 'end'
        assert len(current_path) > 0 and current_path[-1] is elem
        current_path.pop()
        parent = current_path[-1] if len(current_path) > 0 else None
        # `elem` is the current element (fully built),
        # `parent` is its parent (some of its children after `elem`
        # might not have been parsed yet)
        #
        # ... do something ...

解决方案 9:

如果您使用 lxml,我可以使用以下命令获取父元素:

parent_node = next(child_node.iterancestors())

如果元素没有祖先,则会引发StopIteration异常 - 因此,如果您遇到这种情况,请做好捕获该异常的准备。

解决方案 10:

import xml.etree.ElementTree as ET

f1 = "yourFile"

xmlTree = ET.parse(f1)

for root in xmlTree.getroot():
    print(root.tag)

解决方案 11:

另一种方法是如果只想要一个子元素的父元素并且也知道子元素的 xpath。

parentElement = subElement.find(xpath+"/..")

解决方案 12:

查看 19.7.2.2. 部分:支持的 XPath 语法...

使用路径查找节点的父节点:

parent_node = node.find('..')
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1325  
  IPD(Integrated Product Development)流程作为一种先进的产品开发管理模式,在众多企业中得到了广泛应用。它涵盖了从产品概念产生到产品退市的整个生命周期,通过整合跨部门团队、优化流程等方式,显著提升产品开发的效率和质量,进而为项目的成功奠定坚实基础。深入探究IPD流程的五个阶段与项目成功之间...
IPD流程分为几个阶段   4  
  华为作为全球知名的科技企业,其成功背后的管理体系备受关注。IPD(集成产品开发)流程作为华为核心的产品开发管理模式,其中的创新管理与实践更是蕴含着丰富的经验和深刻的智慧,对众多企业具有重要的借鉴意义。IPD流程的核心架构IPD流程旨在打破部门墙,实现跨部门的高效协作,将产品开发视为一个整体的流程。它涵盖了从市场需求分析...
华为IPD是什么   3  
  IPD(Integrated Product Development)研发管理体系作为一种先进的产品开发模式,在众多企业的发展历程中发挥了至关重要的作用。它不仅仅是一套流程,更是一种理念,一种能够全方位提升企业竞争力,推动企业持续发展的有效工具。深入探究IPD研发管理体系如何助力企业持续发展,对于众多渴望在市场中立足并...
IPD管理流程   3  
  IPD(Integrated Product Development)流程管理旨在通过整合产品开发流程、团队和资源,实现产品的快速、高质量交付。在这一过程中,有效降低成本是企业提升竞争力的关键。通过优化IPD流程管理中的各个环节,可以在不牺牲产品质量和性能的前提下,实现成本的显著降低,为企业创造更大的价值。优化产品规划...
IPD流程分为几个阶段   4  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用