21

有没有办法让 defaultdict 默认返回密钥?或者一些具有等效行为的数据结构?即,在初始化字典之后d

>>> d['a'] = 1
>>> d['a']
1
>>> d['b']
'b'
>>> d['c']
'c'

我只看到默认字典采用不带参数的函数,所以我不确定除了创建一种新的字典之外是否还有其他解决方案。

4

4 回答 4

25

我会覆盖以下__missing__方法dict

>>> class MyDefaultDict(dict):
...     def __missing__(self, key):
...         self[key] = key
...         return key
...
>>> d = MyDefaultDict()
>>> d['joe']
'joe'
>>> d
{'joe': 'joe'}
于 2012-05-29T00:28:39.790 回答
7

编辑: 糟糕,我刚刚意识到我文件中的代码最初来自另一个 stackoverflow 答案!https://stackoverflow.com/a/2912455/456876,去投票。

这就是我使用的 - 它是一个 defaultdict 变体,它将键作为参数传递给默认值工厂函数,该函数作为参数传递给init,而不是没有参数:

class keybased_defaultdict(defaultdict):
    def __missing__(self, key):
        if self.default_factory is None:
            raise KeyError(key)
        else:
            value = self[key] = self.default_factory(key)
            return value

这是您想要的用途:

>>> d = keybased_defaultdict(lambda x: x)
>>> d[1]
1
>>> d['a']
'a'

其他可能性:

>>> d = keybased_defaultdict(lambda x: len(x))
>>> d['a']
1
>>> d['abc']
3
于 2012-05-29T00:36:54.487 回答
4

如果你不想继承 dict,你可以尝试使用

d.get('a', 'a')
d.get('b', 'b')
d.get('c', 'c')

我认为为此目的更清晰,更不神奇

如果你是一个 DRY 狂热者并且只有单个字符键,你可以这样做:)

d.get(*'a'*2)
d.get(*'b'*2)
d.get(*'c'*2)
于 2012-05-29T00:59:32.027 回答
3

您可能必须编写自己的类来继承(或类似)defaultdict并覆盖该__getitem__方法。

于 2012-05-29T00:27:30.973 回答