22

如果“车库”中只有一辆“汽车”,则以下代码会产生错误:

import xmltodict

mydict = xmltodict.parse(xmlstringResults)    
for carsInGarage in mydict['garage']['car']:
    # do something...

原因是mydict['garage']['car']如果“汽车”有多个元素,那只是一个列表。所以我做了这样的事情:

import xmltodict

mydict = xmltodict.parse(xmlstringResults)
if isinstance(mydict['garage']['car'], list):
    for carsInGarage in mydict['garage']['car']:
        # do something for each car...
else:
    # do something for the car

让代码运行。但对于更高级的操作,这不是解决方案。

是否有人知道使用某种功能,即使只有一个元素?

4

4 回答 4

32

这个问题在 Github 上的这个 issue中讨论过。xmltodict 包现在支持

d = xmltodict.parse(s, force_list={'car'})

尽管如果该字段不存在,这仍然不会创建一个空列表。

于 2016-08-21T20:07:18.597 回答
4

这当然不是一种优雅的方式,但这是我为运行代码所做的工作(如果有人遇到同样的问题并通过谷歌找到了这个):

import xmltodict

def guaranteed_list(x):
    if not x:
        return []
    elif isinstance(x, list):
        return x
    else:
        return [x]

mydict = xmltodict.parse(xmlstringResults)    
for carsInGarage in guaranteed_list(mydict['garage']['car']):
    # do something...

但我想我会再次编写我的代码并“直接使用 XML”作为评论之一。

于 2016-05-19T09:42:02.487 回答
2

我正在使用的组合

1)

json_dict = xmltodict.parse(s, force_list={'item'})

2)

# Removes a level in python dict if it has only one specific key
# 
# Examples: 
# recursive_skip_dict_key_level({"c": {"a": "b"}}, "c")  # -> {"a", "b"}
# recursive_skip_dict_key_level({"c": ["a", "b"]}, "c") # -> ["a", "b"]
#
def recursive_skip_dict_key_level(d, skipped_key):
    if issubclass(type(d), dict):
        if list(d.keys()) == [skipped_key]:
            return recursive_skip_dict_key_level(d[skipped_key], skipped_key)
        else:
            for key in d.keys():
                d[key] = recursive_skip_dict_key_level(d[key], skipped_key)
            return d
    elif issubclass(type(d), list):
        new_list = []
        for e in d: 
            new_list.append(recursive_skip_dict_key_level(e, skipped_key))
        return new_list
    else: 
        return d

# Removes None values from a dict
# 
# Examples: 
# recursive_remove_none({"a": None})  # -> {}
# recursive_remove_none([None]) # -> []
#
def recursive_remove_none(d):
    if issubclass(type(d), dict):
        new_dict = {}
        for key in d.keys():
            if not (d[key] is None):
                new_dict[key] = recursive_remove_none(d[key])
        return new_dict
    elif issubclass(type(d), list):
        new_list = []
        for e in d: 
            if not (e is None):
                new_list.append(recursive_remove_none(e))
        return new_list
    else: 
        return d       

json_dict = recursive_skip_dict_key_level(json_dict, "item")
json_dict = recursive_remove_none(json_dict)

将任何“项目”XML 元素解释为列表。

于 2020-11-06T03:38:58.787 回答
1

除了现有的答案之外,xmltodict 现在还支持以下内容以强制所有内容成为列表:

xml = xmltodict.parse(s, force_list=True)
于 2021-07-08T12:54:24.897 回答