我在包含xml的数据库中有很多行,我正在尝试编写一个Python脚本,该脚本将遍历这些行并计算特定节点属性的实例数量.例如,我的树看起来像:
<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>
Run Code Online (Sandbox Code Playgroud)
如何使用Python访问XML中的属性1和2?
Ale*_*lli 721
我建议ElementTree
.同一API的其他兼容实现,例如lxml
,以及cElementTree
Python标准库本身; 但是,在这种情况下,他们主要添加的是更快的速度 - 编程部分的简易性取决于ElementTree
定义的API .
root
从XML 构建Element实例后,例如使用XML函数,或通过解析类似的文件
import xml.etree.ElementTree as ET
root = ET.parse('thefile.xml').getroot()
Run Code Online (Sandbox Code Playgroud)
或者显示的任何其他方式ElementTree
,您只需执行以下操作:
for type_tag in root.findall('bar/type'):
value = type_tag.get('foobar')
print(value)
Run Code Online (Sandbox Code Playgroud)
和类似的,通常非常简单的代码模式.
Rya*_*sen 415
minidom
是最快捷,最直接的:
XML:
<data>
<items>
<item name="item1"></item>
<item name="item2"></item>
<item name="item3"></item>
<item name="item4"></item>
</items>
</data>
Run Code Online (Sandbox Code Playgroud)
蟒蛇:
from xml.dom import minidom
xmldoc = minidom.parse('items.xml')
itemlist = xmldoc.getElementsByTagName('item')
print(len(itemlist))
print(itemlist[0].attributes['name'].value)
for s in itemlist:
print(s.attributes['name'].value)
Run Code Online (Sandbox Code Playgroud)
OUTPUT
4
item1
item1
item2
item3
item4
Run Code Online (Sandbox Code Playgroud)
YOU*_*YOU 229
你可以使用BeautifulSoup
from bs4 import BeautifulSoup
x="""<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>"""
y=BeautifulSoup(x)
>>> y.foo.bar.type["foobar"]
u'1'
>>> y.foo.bar.findAll("type")
[<type foobar="1"></type>, <type foobar="2"></type>]
>>> y.foo.bar.findAll("type")[0]["foobar"]
u'1'
>>> y.foo.bar.findAll("type")[1]["foobar"]
u'2'
Run Code Online (Sandbox Code Playgroud)
Cyr*_*rus 90
那里有很多选择.如果速度和内存使用是一个问题,cElementTree看起来很棒.与简单地使用文件读取相比,它的开销非常小readlines
.
相关指标可在下表中找到,从cElementTree网站复制:
library time space
xml.dom.minidom (Python 2.1) 6.3 s 80000K
gnosis.objectify 2.0 s 22000k
xml.dom.minidom (Python 2.4) 1.4 s 53000k
ElementTree 1.2 1.6 s 14500k
ElementTree 1.2.4/1.3 1.1 s 14500k
cDomlette (C extension) 0.540 s 20500k
PyRXPU (C extension) 0.175 s 10850k
libxml2 (C extension) 0.098 s 16000k
readlines (read as utf-8) 0.093 s 8850k
cElementTree (C extension) --> 0.047 s 4900K <--
readlines (read as ascii) 0.032 s 5050k
Run Code Online (Sandbox Code Playgroud)
正如@jfs所指出的那样,cElementTree
它与Python捆绑在一起:
from xml.etree import cElementTree as ElementTree
.from xml.etree import ElementTree
自动使用加速C版本).myi*_*rim 41
为简单起见,我建议使用xmltodict.
它将您的xml解析为OrderedDict;
>>> e = '<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo> '
>>> import xmltodict
>>> result = xmltodict.parse(e)
>>> result
OrderedDict([(u'foo', OrderedDict([(u'bar', OrderedDict([(u'type', [OrderedDict([(u'@foobar', u'1')]), OrderedDict([(u'@foobar', u'2')])])]))]))])
>>> result['foo']
OrderedDict([(u'bar', OrderedDict([(u'type', [OrderedDict([(u'@foobar', u'1')]), OrderedDict([(u'@foobar', u'2')])])]))])
>>> result['foo']['bar']
OrderedDict([(u'type', [OrderedDict([(u'@foobar', u'1')]), OrderedDict([(u'@foobar', u'2')])])])
Run Code Online (Sandbox Code Playgroud)
Rya*_*rom 37
lxml.objectify非常简单.
以您的示例文本为例:
from lxml import objectify
from collections import defaultdict
count = defaultdict(int)
root = objectify.fromstring(text)
for item in root.bar.type:
count[item.attrib.get("foobar")] += 1
print dict(count)
Run Code Online (Sandbox Code Playgroud)
输出:
{'1': 1, '2': 1}
Run Code Online (Sandbox Code Playgroud)
Tor*_*amo 20
Python有一个expat xml解析器的接口.
xml.parsers.expat
Run Code Online (Sandbox Code Playgroud)
它是一个非验证解析器,因此不会捕获到错误的xml.但是如果你知道你的文件是正确的,那么这是非常好的,你可能会得到你想要的确切信息,你可以随时丢弃其余的.
stringofxml = """<foo>
<bar>
<type arg="value" />
<type arg="value" />
<type arg="value" />
</bar>
<bar>
<type arg="value" />
</bar>
</foo>"""
count = 0
def start(name, attr):
global count
if name == 'type':
count += 1
p = expat.ParserCreate()
p.StartElementHandler = start
p.Parse(stringofxml)
print count # prints 4
Run Code Online (Sandbox Code Playgroud)
jch*_*ger 13
只是为了增加另一种可能性,你可以使用untangle,因为它是一个简单的xml-to-python-object库.这里有一个例子:
安装
pip install untangle
Run Code Online (Sandbox Code Playgroud)
用法
你的xml文件(稍有改动):
<foo>
<bar name="bar_name">
<type foobar="1"/>
</bar>
</foo>
Run Code Online (Sandbox Code Playgroud)
使用untangle访问属性:
import untangle
obj = untangle.parse('/path_to_xml_file/file.xml')
print obj.foo.bar['name']
print obj.foo.bar.type['foobar']
Run Code Online (Sandbox Code Playgroud)
输出将是:
bar_name
1
Run Code Online (Sandbox Code Playgroud)
有关untangle的更多信息,请点击此处.
同时,(如果你很好奇),你可以找到的工具列表与XML和Python的工作在这里(你也将看到最常见的是由以前的答案中提到).
小智 13
我可能会建议使用declxml.
完全公开:我编写了这个库,因为我正在寻找一种在XML和Python数据结构之间进行转换的方法,而无需使用ElementTree编写数十行命令式解析/序列化代码.
使用declxml,您可以使用处理器以声明方式定义XML文档的结构以及如何在XML和Python数据结构之间进行映射.处理器既可用于序列化和解析,也可用于基本级别的验证.
解析Python数据结构很简单:
import declxml as xml
xml_string = """
<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>
"""
processor = xml.dictionary('foo', [
xml.dictionary('bar', [
xml.array(xml.integer('type', attribute='foobar'))
])
])
xml.parse_from_string(processor, xml_string)
Run Code Online (Sandbox Code Playgroud)
产生输出:
{'bar': {'foobar': [1, 2]}}
Run Code Online (Sandbox Code Playgroud)
您还可以使用相同的处理器将数据序列化为XML
data = {'bar': {
'foobar': [7, 3, 21, 16, 11]
}}
xml.serialize_to_string(processor, data, indent=' ')
Run Code Online (Sandbox Code Playgroud)
这产生以下输出
<?xml version="1.0" ?>
<foo>
<bar>
<type foobar="7"/>
<type foobar="3"/>
<type foobar="21"/>
<type foobar="16"/>
<type foobar="11"/>
</bar>
</foo>
Run Code Online (Sandbox Code Playgroud)
如果要使用对象而不是字典,则可以定义处理器以将数据转换为对象以及从对象转换数据.
import declxml as xml
class Bar:
def __init__(self):
self.foobars = []
def __repr__(self):
return 'Bar(foobars={})'.format(self.foobars)
xml_string = """
<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>
"""
processor = xml.dictionary('foo', [
xml.user_object('bar', Bar, [
xml.array(xml.integer('type', attribute='foobar'), alias='foobars')
])
])
xml.parse_from_string(processor, xml_string)
Run Code Online (Sandbox Code Playgroud)
这产生以下输出
{'bar': Bar(foobars=[1, 2])}
Run Code Online (Sandbox Code Playgroud)
Jan*_*ila 10
这里有一个非常简单但有效的代码使用cElementTree
.
try:
import cElementTree as ET
except ImportError:
try:
# Python 2.5 need to import a different module
import xml.etree.cElementTree as ET
except ImportError:
exit_err("Failed to import cElementTree from any known place")
def find_in_tree(tree, node):
found = tree.find(node)
if found == None:
print "No %s in file" % node
found = []
return found
# Parse a xml file (specify the path)
def_file = "xml_file_name.xml"
try:
dom = ET.parse(open(def_file, "r"))
root = dom.getroot()
except:
exit_err("Unable to open and parse input definition file: " + def_file)
# Parse to find the child nodes list of node 'myNode'
fwdefs = find_in_tree(root,"myNode")
Run Code Online (Sandbox Code Playgroud)
资源:
http://www.snip2code.com/Snippet/991/python-xml-parse?fromPage=1
Ahi*_*ito 10
XML:
<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>
Run Code Online (Sandbox Code Playgroud)
蟒蛇代码:
import xml.etree.cElementTree as ET
tree = ET.parse("foo.xml")
root = tree.getroot()
root_tag = root.tag
print(root_tag)
for form in root.findall("./bar/type"):
x=(form.attrib)
z=list(x)
for i in z:
print(x[i])
Run Code Online (Sandbox Code Playgroud)
输出:
foo
1
2
Run Code Online (Sandbox Code Playgroud)
有没有必要使用一个lib特定的API,如果你使用python-benedict
。只需从您的 XML 初始化一个新实例并轻松管理它,因为它是一个dict
子类。
安装很简单: pip install python-benedict
from benedict import benedict as bdict
# data-source can be an url, a filepath or data-string (as in this example)
data_source = """
<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>"""
data = bdict.from_xml(data_source)
t_list = data['foo.bar'] # yes, keypath supported
for t in t_list:
print(t['@foobar'])
Run Code Online (Sandbox Code Playgroud)
它支持和标准化的I / O操作多种格式:Base64
,CSV
,JSON
,TOML
,XML
,YAML
和query-string
。
它在GitHub 上经过良好测试和开源。披露:我是作者。
import xml.etree.ElementTree as ET
data = '''<foo>
<bar>
<type foobar="1"/>
<type foobar="2"/>
</bar>
</foo>'''
tree = ET.fromstring(data)
lst = tree.findall('bar/type')
for item in lst:
print item.get('foobar')
Run Code Online (Sandbox Code Playgroud)
这将打印foobar属性的值.
这些是我在使用它们之间进行选择之前会了解的两个最常用的库的优点。
standalone="no"
吗?.node
。