默认情况下,使用 objectify
库的 lxml
API 为元素设置值,将自动检测到的 pytype
分配给该元素和所需的命名空间。
例如,设置根元素:
root = objectify.Element('root')
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
输出:
<root xmlns:py="http://codespeak.net/lxml/objectify/pytype"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" py:pytype="TREE"/>
或为子元素设置一个值:
child = objectify.SubElement(root, 'child')
root.child = 'value'
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
输出:
<root xmlns:py="http://codespeak.net/lxml/objectify/pytype"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" py:pytype="TREE">
<child py:pytype="str">value</child>
</root>
即使使用 ObjectPath 的 setattr:
path = objectify.ObjectPath('root.vader.son')
path.setattr(root, 'Luke')
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
输出:
<root xmlns:py="http://codespeak.net/lxml/objectify/pytype"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" py:pytype="TREE">
<child py:pytype="str">value</child>
<vader>
<son py:pytype="str">Luke</son>
</vader>
</root>
有一些解决方案可以在元素创建后使用
pytype
函数(例如 When using lxml, can the XML be rendered without namespace attributes? 、 Remove "xmlns:py..." with lxml.objectify )删除 deannotate()
及其命名空间。没有任何解决方案可以从一开始就创建没有 pytype
及其命名空间的元素。关于如何做到这一点的任何想法? 最佳答案
在 lxml.objectify
中,有两种元素:由 Element
工厂创建的树元素和由 DataElement
工厂或特定数据类创建的数据元素,例如 StringElement
、 IntElement
(有关更多信息,请参阅 here )。一种解决方案可能是清空特定元素的命名空间和 _pytype
参数,方法是将其分配给空字符串,并且从不使用直接赋值。要从文字创建元素,您必须使用 DataElement 工厂。请注意,如果您有任何特定的命名空间,则必须将命名空间映射而不是空字符串分配给 nsmap 参数。但是有一个问题。如果要创建树元素,将 nsmap
和 _pytype
设置为空字符串,则不会删除命名空间和 pytype。我不知道为什么。所以这个解决方案只适用于数据元素。
这是您尝试构建的树的代码:
root = objectify.Element('root', nsmap='', _pytype='')
# sub elements do not need nsmap or _pytype to be emptied
child = objectify.SubElement(root, 'child')
root.child = objectify.DataElement('value', nsmap='', _pytype='')
path = objectify.ObjectPath('root.vader.son')
path.setattr(root, objectify.DataElement('Luke', nsmap='', _pytype=''))
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
哪些输出:
<root xmlns:py="http://codespeak.net/lxml/objectify/pytype" py:pytype="">
<child>value</child>
<vader>
<son>Luke</son>
</vader>
</root>
不是我们想要的!
解决方案在于使用
ElementMaker
工厂的解决方法。# Create your ElementMaker factory, without annotations.
E = objectify.ElementMaker(annotate=False)
# If you have any namespaces you want to use, assign them to the nsmap
# parameter and assign the default namespace to the namespace parameter.
# E = objectify.ElementMaker(annotate=False, namespace=namespace, nsmap=nsmap)
root = E.root()
print(etree.tostring(root, pretty_print=True))
这输出:
<root/>
引入到树元素的命名空间和 pytype 问题已经解决。现在我们可以分配子元素或数据元素:
objectify.SubElement(root, 'child')
root.child = objectify.DataElement('value', nsmap='', _pytype='')
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
输出:
<root>
<child>value</child>
</root>
使用
setattr()
方法的一个例子是:root = E.root()
path = objectify.ObjectPath('root.vader.son')
path.setattr(root, objectify.DataElement('Luke', nsmap='', _pytype=''))
# mysteriously, the below line works the same as the above line:
# path.setattr(root, E.whatevername('Luke'))
print(etree.tostring(root, pretty_print=True).decode('utf-8'))
其输出为:
<root>
<vader>
<son>Luke</son>
</vader>
</root>
关于python - 在没有 pytype 的情况下设置值 - lxml objectify,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/49478875/