Python中将字典转换为XML以及相关的命名空间解析
尽管xml.etree.ElementTree库通常用来做解析工作,其实它也可以创建XML文档。例如,考虑如下这个函数:
fromxml.etree.ElementTreeimportElement defdict_to_xml(tag,d): ''' Turnasimpledictofkey/valuepairsintoXML ''' elem=Element(tag) forkey,valind.items(): child=Element(key) child.text=str(val) elem.append(child) returnelem
下面是一个使用例子:
>>>s={'name':'GOOG','shares':100,'price':490.1} >>>e=dict_to_xml('stock',s) >>>e <Element'stock'at0x1004b64c8> >>>
转换结果是一个Element实例。对于I/O操作,使用xml.etree.ElementTree中的tostring()函数很容易就能将它转换成一个字节字符串。例如:
>>>fromxml.etree.ElementTreeimporttostring >>>tostring(e) b'<stock><price>490.1</price><shares>100</shares><name>GOOG</name></stock>' >>>
如果你想给某个元素添加属性值,可以使用set()方法:
>>>e.set('_id','1234') >>>tostring(e) b'<stock_id="1234"><price>490.1</price><shares>100</shares><name>GOOG</name> </stock>' >>>
如果你还想保持元素的顺序,可以考虑构造一个OrderedDict来代替一个普通的字典。当创建XML的时候,你被限制只能构造字符串类型的值。例如:
defdict_to_xml_str(tag,d): ''' Turnasimpledictofkey/valuepairsintoXML ''' parts=['<{}>'.format(tag)] forkey,valind.items(): parts.append('<{0}>{1}</{0}>'.format(key,val)) parts.append('</{}>'.format(tag)) return''.join(parts)
问题是如果你手动的去构造的时候可能会碰到一些麻烦。例如,当字典的值中包含一些特殊字符的时候会怎样呢?
>>>d={'name':'<spam>'} >>>#Stringcreation >>>dict_to_xml_str('item',d) '<item><name><spam></name></item>' >>>#ProperXMLcreation >>>e=dict_to_xml('item',d) >>>tostring(e) b'<item><name><spam></name></item>' >>>
注意到程序的后面那个例子中,字符‘<'和‘>'被替换成了<和>
下面仅供参考,如果你需要手动去转换这些字符,可以使用xml.sax.saxutils中的escape()和unescape()函数。例如:
>>>fromxml.sax.saxutilsimportescape,unescape >>>escape('<spam>') '<spam>' >>>unescape(_) '<spam>' >>>
除了能创建正确的输出外,还有另外一个原因推荐你创建Element实例而不是字符串,那就是使用字符串组合构造一个更大的文档并不是那么容易。而Element实例可以不用考虑解析XML文本的情况下通过多种方式被处理。也就是说,你可以在一个高级数据结构上完成你所有的操作,并在最后以字符串的形式将其输出。
利用命名空间解析XML文档
如果你解析这个文档并执行普通的查询,你会发现这个并不是那么容易,因为所有步骤都变得相当的繁琐。
>>>#Somequeriesthatwork >>>doc.findtext('author') 'DavidBeazley' >>>doc.find('content') <Element'content'at0x100776ec0> >>>#Aqueryinvolvinganamespace(doesn'twork) >>>doc.find('content/html') >>>#Worksiffullyqualified >>>doc.find('content/{http://www.w3.org/1999/xhtml}html') <Element'{http://www.w3.org/1999/xhtml}html'at0x1007767e0> >>>#Doesn'twork >>>doc.findtext('content/{http://www.w3.org/1999/xhtml}html/head/title') >>>#Fullyqualified >>>doc.findtext('content/{http://www.w3.org/1999/xhtml}html/' ...'{http://www.w3.org/1999/xhtml}head/{http://www.w3.org/1999/xhtml}title') 'HelloWorld' >>>
你可以通过将命名空间处理逻辑包装为一个工具类来简化这个过程:
classXMLNamespaces: def__init__(self,**kwargs): self.namespaces={} forname,uriinkwargs.items(): self.register(name,uri) defregister(self,name,uri): self.namespaces[name]='{'+uri+'}' def__call__(self,path): returnpath.format_map(self.namespaces)
通过下面的方式使用这个类:
>>>ns=XMLNamespaces(html='http://www.w3.org/1999/xhtml') >>>doc.find(ns('content/{html}html')) <Element'{http://www.w3.org/1999/xhtml}html'at0x1007767e0> >>>doc.findtext(ns('content/{html}html/{html}head/{html}title')) 'HelloWorld' >>>
讨论
解析含有命名空间的XML文档会比较繁琐。上面的XMLNamespaces仅仅是允许你使用缩略名代替完整的URI将其变得稍微简洁一点。
很不幸的是,在基本的ElementTree解析中没有任何途径获取命名空间的信息。但是,如果你使用iterparse()函数的话就可以获取更多关于命名空间处理范围的信息。例如:
>>>fromxml.etree.ElementTreeimportiterparse >>>forevt,eleminiterparse('ns2.xml',('end','start-ns','end-ns')): ...print(evt,elem) ... end<Element'author'at0x10110de10> start-ns('','http://www.w3.org/1999/xhtml') end<Element'{http://www.w3.org/1999/xhtml}title'at0x1011131b0> end<Element'{http://www.w3.org/1999/xhtml}head'at0x1011130a8> end<Element'{http://www.w3.org/1999/xhtml}h1'at0x101113310> end<Element'{http://www.w3.org/1999/xhtml}body'at0x101113260> end<Element'{http://www.w3.org/1999/xhtml}html'at0x10110df70> end-nsNone end<Element'content'at0x10110de68> end<Element'top'at0x10110dd60> >>>elem#Thisisthetopmostelement <Element'top'at0x10110dd60> >>>
最后一点,如果你要处理的XML文本除了要使用到其他高级XML特性外,还要使用到命名空间,建议你最好是使用lxml函数库来代替ElementTree。例如,lxml对利用DTD验证文档、更好的XPath支持和一些其他高级XML特性等都提供了更好的支持。这一小节其实只是教你如何让XML解析稍微简单一点。