最近遇到了一个继承 Python 内建结构的坑儿和大家分享下。从 Python 2.2 开始,Python 支持继承 Python 内建结构,如 list、dict。最近在实现一个功能,为了简化内容,我直接继承了 dict,但是结果和预期不一样。举个例子:

In : class NewDict(dict):
...:     def __getitem__(self, key):
...:         return 42
...:
In : d = NewDict(a=1)
In : d
Out: {'a': 42}
In : d2 = {}
In : d2.update(d)
In : d2
Out: {'a': 1}

也就是说 NewDict 的__getitem__方法被 dict.update 给忽略了。这让我很惊讶,我之前用 UserDict 的时候是正常的(这次为啥直接用 dict 也不知道抽了什么筋):

In : from UserDict import UserDict

In : class NewDict(UserDict):
...:     def __getitem__(self, key):
...:         return 42
...:

In : d = NewDict(a=1)
In : d['b'] =2
In : d
Out: {'a': 1, 'b': 2}
In : d['b']
Out: 42
In : d2 = {}
In : d2.update(d)
In : d2
Out: {'a': 42, 'b': 42}

这才是对的呀。所以我开始研究找答案。后来在 PyPy 的文档中发现了 原因 ,也就是这种 C 实现的结构的内建方法大部分会忽略重载的那个方法。

之前我以为 UserDict 这样的类是历史遗留问题,现在才知道是有原因的。原来 UserDict、UserString、UserList 这样的模块是非常必要的。