【发布时间】:2016-02-21 04:42:30
【问题描述】:
在 Python 中从值 == None 的字典中删除键的正确方法是什么?
【问题讨论】:
-
您必须澄清您的意思,因为字典中的每个键都必须有一个值,即使该值是
''或0或None。 -
{k: v for k, v in original.items() if v is not None}
标签: python dictionary
在 Python 中从值 == None 的字典中删除键的正确方法是什么?
【问题讨论】:
'' 或 0 或 None。
{k: v for k, v in original.items() if v is not None}
标签: python dictionary
通常,您将通过过滤旧的dict 来创建一个新的dict。字典推导非常适合这种事情:
{k: v for k, v in original.items() if v is not None}
如果你必须更新原来的dict,你可以这样做...
filtered = {k: v for k, v in original.items() if v is not None}
original.clear()
original.update(filtered)
这可能是我能想到的最“干净”的就地删除它们的方法(在迭代时修改 dict 是不安全的)
在python2.x上使用original.iteritems()
【讨论】:
dict.keys 或 dict.items?
dict 总是不安全的。在 python2.x 上迭代 dict.keys() 和 dict.items() 很好,但在 python3.x 上不安全。例如,请参阅stackoverflow.com/a/6777632/748858。
如果您需要递归删除 None 值,最好使用这个:
def delete_none(_dict):
"""Delete None values recursively from all of the dictionaries"""
for key, value in list(_dict.items()):
if isinstance(value, dict):
delete_none(value)
elif value is None:
del _dict[key]
elif isinstance(value, list):
for v_i in value:
if isinstance(v_i, dict):
delete_none(v_i)
return _dict
根据@dave-cz 的建议,添加了支持列表类型值的功能。
@mandragor 添加了额外的 if 语句以允许包含简单列表的字典。
如果您需要从字典、列表、元组、集合中删除所有 None 值,这也是一种解决方案:
def delete_none(_dict):
"""Delete None values recursively from all of the dictionaries, tuples, lists, sets"""
if isinstance(_dict, dict):
for key, value in list(_dict.items()):
if isinstance(value, (list, dict, tuple, set)):
_dict[key] = delete_none(value)
elif value is None or key is None:
del _dict[key]
elif isinstance(_dict, (list, set, tuple)):
_dict = type(_dict)(delete_none(item) for item in _dict if item is not None)
return _dict
passed:
a = {
"a": 12, "b": 34, "c": None,
"k": {"d": 34, "t": None, "m": [{"k": 23, "t": None},[None, 1, 2, 3],{1, 2, None}], None: 123}
}
returned:
a = {
"a": 12, "b": 34,
"k": {"d": 34, "m": [{"k": 23}, [1, 2, 3], {1, 2}]}
}
【讨论】:
_dict.items() 之前添加list() 以避免由于dict 在迭代期间更改大小而引发RunTimeError
elif isinstance(value, list): for v_i in value: delete_none(v_i)
对于python 2.x:
dict((k, v) for k, v in original.items() if v is not None)
您还可以复制dict 以避免在更改原始dict 时对其进行迭代。
for k, v in dict(d).items():
if v is None:
del d[k]
但这对于较大的字典可能不是一个好主意。
【讨论】:
如果你不想复制
for k,v in list(foo.items()):
if v is None:
del foo[k]
【讨论】:
foo.pop(k) 更好,否则会失败
list 关键字创建键/值对的(浅)副本。
Python3递归版本
def drop_nones_inplace(d: dict) -> dict:
"""Recursively drop Nones in dict d in-place and return original dict"""
dd = drop_nones(d)
d.clear()
d.update(dd)
return d
def drop_nones(d: dict) -> dict:
"""Recursively drop Nones in dict d and return a new dict"""
dd = {}
for k, v in d.items():
if isinstance(v, dict):
dd[k] = drop_nones(v)
elif isinstance(v, (list, set, tuple)):
# note: Nones in lists are not dropped
# simply add "if vv is not None" at the end if required
dd[k] = type(v)(drop_nones(vv) if isinstance(vv, dict) else vv
for vv in v)
elif v is not None:
dd[k] = v
return dd
【讨论】:
也许你会发现它很有用:
def clear_dict(d):
if d is None:
return None
elif isinstance(d, list):
return list(filter(lambda x: x is not None, map(clear_dict, d)))
elif not isinstance(d, dict):
return d
else:
r = dict(
filter(lambda x: x[1] is not None,
map(lambda x: (x[0], clear_dict(x[1])),
d.items())))
if not bool(r):
return None
return r
它会:
clear_dict(
{'a': 'b', 'c': {'d': [{'e': None}, {'f': 'g', 'h': None}]}}
)
->
{'a': 'b', 'c': {'d': [{'f': 'g'}]}}
【讨论】: