Как преобразовать XML в объект Python?
Мне нужно загрузить XML файл и преобразовать его в объектно-ориентированную структуру Python. Я хочу принять это:
<main>
<object1 attr="name">content</object>
</main>
И превратите его в нечто вроде этого:
main
main.object1 = "content"
main.object1.attr = "name"
Данные XML будут иметь более сложную структуру, чем это, и я не могу жестко закодировать имена элементов. Имена атрибутов необходимо собирать при разборе и использовании в качестве свойств объекта.
Как преобразовать XML-данные в объект Python?
Ответы
Ответ 1
Стоит посмотреть lxml.objectify
.
xml = """<main>
<object1 attr="name">content</object1>
<object1 attr="foo">contenbar</object1>
<test>me</test>
</main>"""
from lxml import objectify
main = objectify.fromstring(xml)
main.object1[0] # content
main.object1[1] # contenbar
main.object1[0].get("attr") # name
main.test # me
Или, наоборот, для создания XML-структур:
item = objectify.Element("item")
item.title = "Best of python"
item.price = 17.98
item.price.set("currency", "EUR")
order = objectify.Element("order")
order.append(item)
order.item.quantity = 3
order.price = sum(item.price * item.quantity for item in order.item)
import lxml.etree
print(lxml.etree.tostring(order, pretty_print=True))
Вывод:
<order>
<item>
<title>Best of python</title>
<price currency="EUR">17.98</price>
<quantity>3</quantity>
</item>
<price>53.94</price>
</order>
Ответ 2
Я рекомендую это более одного раза сегодня, но попробуйте Beautiful Soup (easy_install BeautifulSoup).
from BeautifulSoup import BeautifulSoup
xml = """
<main>
<object attr="name">content</object>
</main>
"""
soup = BeautifulSoup(xml)
# look in the main node for object with attr=name, optionally look up attrs with regex
my_objects = soup.main.findAll("object", attrs={'attr':'name'})
for my_object in my_objects:
# this will print a list of the contents of the tag
print my_object.contents
# if only text is inside the tag you can use this
# print tag.string
Ответ 3
Дэвид Мерц gnosis.xml.objectify, похоже, сделает это за вас. Документация немного сложна, но на ней есть несколько статей IBM, в том числе этот.
from gnosis.xml import objectify
xml = "<root><nodes><node>node 1</node><node>node 2</node></nodes></root>"
root = objectify.make_instance(xml)
print root.nodes.node[0].PCDATA # node 1
print root.nodes.node[1].PCDATA # node 2
Создание xml из объектов таким образом - это другое дело.
Ответ 4
Как насчет этого
http://evanjones.ca/software/simplexmlparse.html
Ответ 5
#@Stephen:
#"can't hardcode the element names, so I need to collect them
#at parse and use them somehow as the object names."
#I don't think thats possible. Instead you can do this.
#this will help you getting any object with a required name.
import BeautifulSoup
class Coll(object):
"""A class which can hold your Foo clas objects
and retrieve them easily when you want
abstracting the storage and retrieval logic
"""
def __init__(self):
self.foos={}
def add(self, fooobj):
self.foos[fooobj.name]=fooobj
def get(self, name):
return self.foos[name]
class Foo(object):
"""The required class
"""
def __init__(self, name, attr1=None, attr2=None):
self.name=name
self.attr1=attr1
self.attr2=attr2
s="""<main>
<object name="somename">
<attr name="attr1">value1</attr>
<attr name="attr2">value2</attr>
</object>
<object name="someothername">
<attr name="attr1">value3</attr>
<attr name="attr2">value4</attr>
</object>
</main>
"""
#
soup=BeautifulSoup.BeautifulSoup(s)
bars=Coll()
for each in soup.findAll('object'):
bar=Foo(each['name'])
attrs=each.findAll('attr')
for attr in attrs:
setattr(bar, attr['name'], attr.renderContents())
bars.add(bar)
#retrieve objects by name
print bars.get('somename').__dict__
print '\n\n', bars.get('someothername').__dict__
Выход
{'attr2': 'value2', 'name': u'somename', 'attr1': 'value1'}
{'attr2': 'value4', 'name': u'someothername', 'attr1': 'value3'}
Ответ 6
Существует три общих синтаксических анализатора XML для python: xml.dom.minidom, elementree и BeautifulSoup.
IMO, BeautifulSoup на сегодняшний день является лучшим.
http://www.crummy.com/software/BeautifulSoup/
Ответ 7
Если поиск кода для генератора кода не работает, вы можете написать свой собственный, который использует XML в качестве входных и выводимых объектов на выбранном вами языке.
Это не очень сложно, однако трехэтапный процесс Parse XML, Generate Code, Compile/Execute Script делает отладку немного сложнее.