如何在Python中解析XML?

ran*_*its 938 python xml

我在包含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,以及cElementTreePython标准库本身; 但是,在这种情况下,他们主要添加的是更快的速度 - 编程部分的简易性取决于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)

和类似的,通常非常简单的代码模式.

  • 您似乎忽略了Python附带的xml.etree.cElementTree,并且在某些方面更快tham lxml("lxml的iterparse()比cET中的稍慢" - 来自lxml作者的电子邮件). (37认同)
  • 似乎ElementTree有一些漏洞问题,这是来自文档的引用:`警告xml.etree.ElementTree模块对于恶意构造的数据是不安全的.如果您需要解析不受信任或未经身份验证的数据,请参阅XML漏洞 (13认同)
  • `lxml`增加了速度.它提供了对父节点,XML源中的行号等信息的轻松访问,这些信息在几种情况下非常有用. (11认同)
  • ElementTree工作并包含在Python中.虽然XPath支持有限,但您无法遍历元素的父级,这会降低开发速度(特别是如果您不知道这一点).有关详细信息,请参阅[python xml query get parent](http://stackoverflow.com/questions/5373902/python-xml-query-get-parent). (7认同)
  • @Cristik大多数xml解析器似乎都是这种情况,请参阅[XML漏洞页面](https://docs.python.org/3.4/library/xml.html#xml-vulnerabilities). (5认同)
  • @paul,*整洁* - 我打算从这里获得灵感(虽然可能在实际的柯南道尔引用:-)未来的第三版"果壳中的Python"(其中我恰好重新开始)写了XML章节以完全覆盖ElementTree - 只是将该章的草稿邮寄给/ effbot以获得他的反馈...... :-). (2认同)
  • 来自[docs](https://docs.python.org/3.5/library/xml.etree.elementtree.html#module-xml.etree.ElementTree):版本3.3中已更改:此模块将在任何时候使用快速实现可用.不推荐使用`xml.etree.cElementTree`模块. (2认同)

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)

  • 我想出来,万一有人有同样的问题.它是s.childNodes [0] .nodeValue (86认同)
  • 你如何获得"item1"的价值?例如:<item name ="item1"> Value1 </ item> (9认同)

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)

  • 三年后与BS4这是一个很好的解决方案,非常灵活,特别是在源没有很好地形成 (44认同)
  • 实际上,BeautifulSoup for XML中有`BeautifulStoneSoup` (7认同)
  • @YOU`BeautifulStoneSoup`很受欢迎.只需使用`BeautifulSoup(source_xml,features ="xml")` (7认同)
  • 另外3年后,我只是尝试使用`ElementTree`加载XML,不幸的是它无法解析,除非我在地方调整源,但是'BeautifulSoup`刚刚工作而没有任何改变! (4认同)
  • @andi您的意思是“已弃用”。“折旧”是指其价值下降,通常是由于使用年限或正常使用引起的磨损。 (2认同)
  • 又过了三年,现在 BS4 还不够快。需要很长时间。寻找更快的解决方案 (2认同)

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捆绑在一起:

  • Python 2 : from xml.etree import cElementTree as ElementTree.
  • Python 3 :( from xml.etree import ElementTree自动使用加速C版本).

  • 使用cElementTree有什么缺点吗?这似乎是一个明智的选择. (9认同)
  • @Stunner:它在stdlib中,即你不需要下载任何东西.在Python 2上:`从xml.etree导入cElementTree作为ElementTree`.在Python 3上:`from xml.etree import ElementTree`(自动使用加速C版本) (8认同)
  • 显然他们不想在OS X上使用该库,因为我花了超过15分钟试图找出从哪里下载它并且没有链接工作.缺乏文件可以阻止好的项目蓬勃发展,希望更多人能够意识到这一点. (6认同)

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)

  • 请记住,OrderedDict不支持重复键.大多数XML都充满了相同类型的多个兄弟(例如,节中的所有段落,或者栏中的所有类型).所以这只适用于非常有限的特殊情况. (3认同)
  • 同意.如果你不需要XPath或任何复杂的东西,这使用起来要简单得多(特别是在解释器中); 方便用于发布XML而不是JSON的REST API (2认同)
  • @TextGeek 在这种情况下,`result["foo"]["bar"]["type"]` 是所有 `&lt;type&gt;` 元素的列表,所以它仍然有效(即使结构可能有点意外)。 (2认同)

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)


Fab*_*amo 8

没有必要使用一个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操作多种格式:Base64CSVJSONTOMLXMLYAMLquery-string

它在GitHub 上经过良好测试和开源。披露:我是作者。


EMP*_*EMP 6

我发现Python xml.domxml.dom.minidom非常简单.请记住,DOM不适合大量的XML,但如果您的输入相当小,那么这将很好.


Sou*_*Dey 5

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属性的值.


G M*_*G M 5

xml.etree.ElementTree与lxml

这些是我在使用它们之间进行选择之前会了解的两个最常用的库的优点。

xml.etree.ElementTree:

  1. 来自标准库:无需安装任何模块

xml文件

  1. 轻松编写XML声明:例如,您需要添加standalone="no"吗?
  2. 印刷精美:无需额外代码即可拥有漂亮的缩进 XML。
  3. 对象化功能:它使您可以像处理普通的Python对象层次结构一样使用XML .node