Подтвердить что ты не робот

Как преобразовать XML в объект Python?

Мне нужно загрузить XML файл и преобразовать его в объектно-ориентированную структуру Python. Я хочу принять это:

<main>
    <object1 attr="name">content</object>
</main>

И превратите его в нечто вроде этого:

main
main.object1 = "content"
main.object1.attr = "name"

Данные XML будут иметь более сложную структуру, чем это, и я не могу жестко закодировать имена элементов. Имена атрибутов необходимо собирать при разборе и использовании в качестве свойств объекта.

Как преобразовать XML-данные в объект Python?

4b9b3361

Ответ 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 из объектов таким образом - это другое дело.

Ответ 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 делает отладку немного сложнее.