从另一个字典创建包含特定键值对的Python字典
我有一些代码,大概是这样的:
d = {'foo': True, 'bar': 42, 'baz': '!'}
a = {'foo': d['foo'], 'bar': d['bar']}
b = {'foo': d['foo'], 'baz': d['baz']}
c = {'bar': d['bar'], 'baz': d['baz']}
我相信一定有更好的方法来表达这个。我其实看过文档,希望字典的 copy 方法能接受要包含在新字典中的键:
# I'd hoped that something like this would work...
a = d.copy('foo', 'bar')
b = d.copy('foo', 'baz')
c = d.copy('bar', 'baz')
我可以为这个目的写一个函数:
copydict = lambda dct, *keys: {key: dct[key] for key in keys}
a = copydict(d, 'foo', 'bar')
b = copydict(d, 'foo', 'baz')
c = copydict(d, 'bar', 'baz')
有没有比上面更好的解决方案呢?
相关问题:
3 个回答
0
正如@BlaXpirit所说,这个解决方案在速度或可读性方面可能不如你的方法好,但你可以尝试这样做:
>>> from operator import itemgetter
>>> d = {'foo': True, 'bar': 42, 'baz': '!'}
>>> keys = ['foo', 'bar']
>>> dict(zip(keys, itemgetter(*keys)(d)))
{'bar': 42, 'foo': True}
3
我唯一想改进的就是用真正的函数定义,而不是用lambda表达式:
def copy_dict(d, *keys):
"""Make a copy of only the `keys` from dictionary `d`."""
return {key: d[key] for key in keys}
处理缺失的键值可能会很有用,但在Python 2中,你不能很好地将可选的关键字参数和*args混合使用,所以你可能需要使用元组作为参数:
def copy_dict(d, keys, default=None):
"""Make a copy of only the `keys` from dictionary `d`.
Use `default` if a key is missing.
"""
return {key: d.get(key, default) for key in keys}
6
我觉得问题中的这个函数是最好的解决方案。
当然,有人可能会发一些更地道的代码,但我相信那样的代码不会更好或更快。遍历一个列表并通过键从字典中获取项目的速度已经是最快的了。
有一个建议是去掉keys参数前面的星号。参数解包会增加不必要的开销。直接把这些键作为一个元组传进去应该没有问题。