Nis*_*ede 4 python xml xml-parsing python-2.7 xml.etree
对于这个xml
<Departments orgID="123" name="xmllist">
<Department>
<orgID>124</orgID>
<name>A</name>
<type>type a</type>
<status>Active</status>
<Department>
<orgID>125</orgID>
<name>B</name>
<type>type b</type>
<status>Active</status>
<Department>
<orgID>126</orgID>
<name>C</name>
<type>type c</type>
<status>Active</status>
</Department>
</Department>
</Department>
<Department>
<orgID>109449</orgID>
<name>D</name>
<type>type d</type>
<status>Active</status>
</Department>
</Departments>
Run Code Online (Sandbox Code Playgroud)
我如何lxml etree在python中使用节点的所有父节点.
预期输出:输入orgid = 126,它将返回所有父母,如,
{'A':124,'B':125,'C':126}
Run Code Online (Sandbox Code Playgroud)
使用lxml和XPath:
>>> s = '''
... <Departments orgID="123" name="xmllist">
... <Department>
... <orgID>124</orgID>
... <name>A</name>
... <type>type a</type>
... <status>Active</status>
... <Department>
... <orgID>125</orgID>
... <name>B</name>
... <type>type b</type>
... <status>Active</status>
... <Department>
... <orgID>126</orgID>
... <name>C</name>
... <type>type c</type>
... <status>Active</status>
... </Department>
... </Department>
... </Department>
... <Department>
... <orgID>109449</orgID>
... <name>D</name>
... <type>type d</type>
... <status>Active</status>
... </Department>
... </Departments>
... '''
Run Code Online (Sandbox Code Playgroud)
使用ancestor-or-selfaxis,你可以找到节点本身,父,祖父母,...
>>> import lxml.etree as ET
>>> root = ET.fromstring(s)
>>> for target in root.xpath('.//Department/orgID[text()="126"]'):
... d = {
... dept.find('name').text: int(dept.find('orgID').text)
... for dept in target.xpath('ancestor-or-self::Department')
... }
... print(d)
...
{'A': 124, 'C': 126, 'B': 125}
Run Code Online (Sandbox Code Playgroud)
使用lxml的iterancestors()方法。
from lxml import etree
doc = etree.fromstring(xml)
rval = {}
for org in doc.xpath('//orgID[text()="126"]'):
for ancestor in org.iterancestors('Department'):
id=ancestor.find('./orgID').text
name=ancestor.find('./name').text
rval[name]=id
print rval
Run Code Online (Sandbox Code Playgroud)
输出:
{'A': '124', 'C': '126', 'B': '125'}
Run Code Online (Sandbox Code Playgroud)
如果您实际上想保留元素的顺序,那么您不能使用字典,因为您无法控制字典中的键顺序。您必须使用 OrderedDict 或仅使用元组数组:
doc = etree.fromstring(xml)
a = []
for org in doc.xpath('//orgID[text()="126"]'):
for ancestor in org.iterancestors():
if ancestor.find('./orgID') is not None:
id=ancestor.find('./orgID').text
name=ancestor.find('./name').text
elif ancestor.get('orgID'):
id=ancestor.get('orgID')
name=ancestor.get('name')
else:
continue
print id,name
a.append((name,id))
print "In order of discovery:\n ", a
print "From root to child\n ", [x for x in reversed(a)]
print "dict keys are not sorted\n ", dict(a)
Run Code Online (Sandbox Code Playgroud)
输出:
126 C
125 B
124 A
123 xmllist
In order of discovery:
[('C', '126'), ('B', '125'), ('A', '124'), ('xmllist', '123')]
From root to child
[('xmllist', '123'), ('A', '124'), ('B', '125'), ('C', '126')]
dict keys are not sorted
{'A': '124', 'xmllist': '123', 'C': '126', 'B': '125'}
Run Code Online (Sandbox Code Playgroud)