
在日常开发中,处理xml数据是一项常见任务。xml.etree.elementtree是python标准库中一个轻量级且高效的xml解析模块。本教程将聚焦于一个具体的场景:从复杂的xml结构中提取<inter>元素的name属性,并根据其内部<work>元素是否包含groups属性,来决定是否将groups的值与name属性进行拼接。例如,如果groups属性存在,我们希望得到inter_name-groupa, inter_name-groupb这样的结果;如果不存在,则只保留inter_name。
为了更好地理解问题,我们先来看一下示例XML的结构:
<abtshop>
<inters>
<inter name="nim_turk" first-day="20230301" historical="20220103" market="multi">
<works>
<work kind="obopay" run="jbs">
<!-- 无 groups 属性 -->
</work>
</works>
</inter>
<inter name="nim_us_m" first-day="20230301" historical="20220103" market="lone">
<works>
<work kind="obopay" run="jbs" groups="groupA,groupB">
<!-- 包含 groups 属性 -->
</work>
<work kind="silkb" run="jbs" groups="groupA,groupB">
<!-- 包含 groups 属性 -->
</work>
</works>
</inter>
</inters>
</abtshop>我们的目标是从<inter>元素中获取name属性。对于包含<work>子元素且该<work>元素带有groups属性的<inter>,我们需要将name与groups中的每个值进行拼接。
xml.etree.ElementTree模块提供了一套直观的API来解析和操作XML文档。
首先,我们需要导入xml.etree.ElementTree并加载XML文件。
立即学习“Python免费学习笔记(深入)”;
import xml.etree.ElementTree as ET
# 假设XML内容保存在 'test_xml.xml' 文件中
tree = ET.parse("test_xml.xml")
root = tree.getroot() # 获取XML文档的根元素或者,如果XML是字符串,可以使用ET.fromstring()。
核心思路是遍历所有相关的<inter>元素,针对每个<inter>元素,首先获取其name属性。然后,在其内部查找是否存在带有groups属性的<work>子元素。根据查找结果,执行不同的拼接逻辑。
我们首先需要定位到所有<inter>元素。由于它们嵌套在<inters>标签下,我们可以先找到<inters>,然后遍历其所有子元素。
result = [] # 用于存储最终结果的列表
# 遍历所有 <inters> 元素(通常只有一个)
for inters_elem in root.findall('.//inters'):
# 遍历 <inters> 下的所有 <inter> 子元素
for inter_elem in inters_elem:
# 获取当前 <inter> 元素的 name 属性
inter_name = inter_elem.get('name')
# ... 后续处理逻辑 ...在获取到inter_name之后,我们需要判断当前inter_elem内部是否有带有groups属性的<work>元素。Element.find()方法非常适合这个场景,因为它会返回第一个匹配的子元素,如果没有找到则返回None。
XPath表达式".//work[@groups]"的含义是:
# 在当前 <inter> 元素内部查找第一个带有 'groups' 属性的 <work> 元素
work_with_groups = inter_elem.find(".//work[@groups]")
if work_with_groups is not None:
# 如果找到了带有 'groups' 属性的 <work> 元素
groups_str = work_with_groups.get('groups') # 获取 groups 属性值,例如 "groupA,groupB"
group_list = groups_str.split(',') # 按逗号分割成列表
# 遍历每个组名,并与 inter_name 拼接
for group_name in group_list:
combined_name = inter_name + '-' + group_name
result.append(combined_name)
else:
# 如果没有找到带有 'groups' 属性的 <work> 元素,则只添加 inter_name
result.append(inter_name)将上述步骤整合起来,形成完整的解决方案:
import xml.etree.ElementTree as ET
# 假设XML内容保存在 'test_xml.xml' 文件中
# 创建一个示例XML文件用于测试
xml_content = """
<abtshop>
<dDirectory>dub</dDirectory>
<S>statusd</S>
<work>worklogs</work>
<custs>
<cust>nim-us</cust>
</custs>
<mileage>999</mileage>
<defaults>
<default type="mercley">
<user>dairy</user>
<exec>slm.sh</exec>
<env>
<var name="SAN_HOME">youyou-11</var>
</env>
</default>
</defaults>
<inters>
<inter name="nim_turk" first-day="20230301" historical="20220103" market="multi">
<works>
<work kind="obopay" run="jbs">
<args>
<arg name="distance">180000</arg>
</args>
</work>
<work kind="silkb" run="jbs">
<args>
<arg name="distance">180000</arg>
</args>
</work>
</works>
</inter>
<inter name="nim_us_m" first-day="20230301" historical="20220103" market="lone">
<works>
<work kind="obopay" run="jbs" groups="groupA,groupB">
<args>
<arg name="distance">120000</arg>
<arg name="jbsopt">xmas_size=1200000</arg>
<arg name="jbsopt">of_obopaying_threads=2</arg>
</args>
</work>
<work kind="silkb" run="jbs" groups="groupA,groupB">
<args>
<arg name="distance">120000</arg>
<arg name="jbsopt">xmas_size=1200000</arg>
</args>
</work>
</works>
</inter>
</inters>
</abtshop>
"""
# 将XML内容写入文件
with open("test_xml.xml", "w", encoding="utf-8") as f:
f.write(xml_content)
# 解析XML文件
root = ET.parse("test_xml.xml").getroot()
result = [] # 存储最终结果的列表
# 遍历所有 <inters> 元素(通常只有一个,但使用 findall 更具通用性)
for inters_elem in root.findall('.//inters'):
# 遍历 <inters> 元素下的所有 <inter> 子元素
for inter_elem in inters_elem:
inter_name = inter_elem.get('name') # 获取 <inter> 元素的 name 属性值
# 在当前 <inter> 元素内部查找第一个带有 'groups' 属性的 <work> 元素
work_with_groups = inter_elem.find(".//work[@groups]")
if work_with_groups is not None:
# 如果找到了带有 'groups' 属性的 <work> 元素
groups_str = work_with_groups.get('groups') # 获取 groups 属性值
group_list = groups_str.split(',') # 按逗号分割成列表
# 遍历每个组名,并与 inter_name 拼接
for group_name in group_list:
combined_name = inter_name + '-' + group_name
result.append(combined_name)
else:
# 如果没有找到带有 'groups' 属性的 <work> 元素,则只添加 inter_name
result.append(inter_name)
print(result)预期输出:
['nim_turk', 'nim_us_m-groupA', 'nim_us_m-groupB']
本教程演示了如何利用Python的xml.etree.ElementTree模块高效地解析XML数据,并根据特定条件(是否存在groups属性)从不同层级的元素中提取和拼接数据。通过灵活运用findall()、find()和get()等方法,结合XPath表达式,我们可以精确地定位目标数据并实现复杂的逻辑处理,从而满足多样化的XML数据处理需求。
以上就是Python ElementTree教程:条件提取XML属性并拼接的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号