0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

__missing__()的实现原理

科技绿洲 来源:Python实用宝典 作者:Python实用宝典 2023-10-30 14:58 次阅读

上篇文章中,我有一个核心的发现:Python 内置类型的特殊方法(含魔术方法与其它方法)由 C 语言独立实现,在 Python 层面不存在调用关系。

但是,文中也提到了一个例外:一个非常神秘的魔术方法。

这个方法非常不起眼,用途狭窄,我几乎从未注意过它,然而,当发现它可能是上述“定律”的唯一例外情况时,我认为值得再写一篇文章来详细审视一下它。

本文主要关注的问题有:

(1) missing ()到底是何方神圣?

(2) missing ()有什么特别之处?擅长“大变活人”魔术?

(3) missing ()是否真的是上述发现的例外?如果是的话,为什么会有这种特例?

1、有点价值的__missing__()

从普通的字典中取值时,可能会出现 key 不存在的情况:

dd = {'name':'PythonCat'}
dd.get('age')        # 结果:None
dd.get('age', 18)    # 结果:18
dd['age']            # 报错 KeyError
dd.__getitem__('age')  # 等同于 dd['age']

图片

对于 get() 方法,它是有返回值的,而且可以传入第二个参数,作为 key 不存在时的返回内容,因此还可以接受。但是,另外两种写法都会报错。

为了解决后两种写法的问题,就可以用到 missing () 魔术方法。

现在,假设我们有一个这样的诉求:从字典中取某个 key 对应的 value,如果有值则返回值,如果没有值则插入 key,并且给它一个默认值(例如一个空列表)。

如果用原生的 dict,并不太好实现,但是,Python 提供了一个非常好用的扩展类collections.defaultdict

图片

如图所示,当取不存在的 key 时,没有再报 KeyError,而是默认存入到字典中。

为什么 defaultdict 可以做到这一点呢?

原因是 defaultdict 在继承了内置类型 dict 之后,还定义了一个 missing () 方法,当 __getitem__取不存在的值时,它就会调用入参中传入的工厂函数(上例是调用 list(),创建空列表)。

作为最典型的示例,defaultdict 在文档注释中写到:

图片

简而言之,** missing ()的主要作用就是由__getitem__在缺失 key 时调用,从而避免出现 KeyError。**

另外一个典型的使用例子是collections.Counter,它也是 dict 的子类,在取未被统计的 key 时,返回计数 0:

图片

2、神出鬼没的__missing__()

由上可知, missing ()在__getitem__()取不到值时会被调用,但是,我不经意间还发现了一个细节:** getitem ()在取不到值时,并不一定会调用__missing__()。**

这是因为它并非内置类型的必要属性,并没有在字典基类中被预先定义。

如果你直接从 dict 类型中取该属性值,会报属性不存在:AttributeError: type object 'object' has no attribute '__missing__'

使用 dir() 查看,发现确实不存在该属性:

图片

如果从 dict 的父类即 object 中查看,也会发现同样的结果。

这是怎么回事呢?为什么在 dict 和 object 中都没有__missing__属性呢?

然而,查阅最新的官方文档,object 中分明包含这个属性:

图片

出处:https://docs.python.org/3/reference/datamodel.html?highlight= missing #object.missing

也就是说,理论上 object 类中会预定义__missing__,其文档证明了这一点,然而实际上它并没有被定义!文档与现实出现了偏差!

如此一来,当 dict 的子类(例如 defaultdict 和 Counter)在定义__missing__ 时,这个魔术方法事实上只属于该子类,也就是说,它是一个诞生于子类中的魔术方法!

据此,我有一个不成熟的猜想: getitem ()会判断当前对象是否是 dict 的子类,且是否拥有__missing__(),然后才会去调用它(如果父类中也有该方法,则不会先作判断,而是直接就调用了)。

我在交流群里说出了这个猜想,有同学很快在 CPython 源码中找到验证:

图片

而这就有意思了, 在内置类型的子类上才存在的魔术方法, 纵观整个 Python 世界,恐怕再难以找出第二例。

我突然有一个联想:这神出鬼没的__missing__(),就像是一个擅长玩“大变活人”的魔术师,先让观众在外面透过玻璃看到他(即官方文档),然而揭开门时,他并不在里面(即内置类型),再变换一下道具,他又完好无损就出现了(即 dict 的子类)。

3、被施魔法的__missing__()

missing () 的神奇之处,除了它本身会变“魔术”之外,它还需要一股强大的“魔法”才能驱动。

上篇文章中,我发现原生的魔术方法间相互独立,它们在 C 语言界面可能有相同的核心逻辑,但是在 Python 语言界面,却并不存在着调用关系:

图片

魔术方法的这种“老死不相往来”的表现,违背了一般的代码复用原则,也是导致内置类型的子类会出现某些奇怪表现的原因。

官方 Python 宁肯提供新的 UserString、UserList、UserDict 子类,也不愿意复用魔术方法,唯一合理的解释似乎是令魔术方法相互调用的代价太大。

但是,对于特例__missing__(),Python 却不得不妥协,不得不付出这种代价!

missing () 是魔术方法的“ 二等公民 ”,它没有独立的调用入口,只能被动地由 getitem () 调用,即__missing__() 依赖于__getitem__()。

不同于那些“ 一等公民 ”,例如 init ()、 enter ()、 len ()、 eq () 等等,它们要么是在对象生命周期或执行过程的某个节点被触发,要么由某个内置函数或操作符触发,这些都是相对独立的事件,无所依赖。

** missing () 依赖于__getitem__(),才能实现方法调用;而 getitem () 也要依赖 missing (),才能实现完整功能。**

为了实现这一点, getitem ()在解释器代码中开了个后门,从 C 语言界面折返回 Python 界面,去调用那个名为“ missing ”的特定方法。

图片

而这就是真正的“魔法”了,目前为止, missing ()似乎是唯一一个享受了此等待遇的魔术方法!

4、小结

Python 的字典提供了两种取值的内置方法,即__getitem__() 和 get(),当取值不存在时,它们的处理策略是不一样的:前者会报错KeyError,而后者会返回 None。

为什么 Python 要提供两个不同的方法呢?或者应该问,为什么 Python 要令这两个方法做出不一样的处理呢?

这可能有一个很复杂(也可能是很简单)的解释,本文暂不深究了。

不过有一点是可以确定的:即原生 dict 类型简单粗暴地抛KeyError的做法有所不足。

为了让字典类型有更强大的表现(或者说让__getitem__()作出 get() 那样的表现),Python 让字典的子类可以定义__missing__(),供__getitem__()查找调用。

本文梳理了__missing__()的实现原理,从而揭示出它并非是一个毫不起眼的存在,恰恰相反,它是唯一一个打破了魔术方法间壁垒,支持被其它魔术方法调用的特例!

Python 为了维持魔术方法的独立性,不惜煞费苦心地引入了 UserString、UserList、UserDict 这些派生类,但是对于 missing (),它却选择了妥协。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    4333

    浏览量

    62723
  • python
    +关注

    关注

    56

    文章

    4797

    浏览量

    84787
  • C 语言
    +关注

    关注

    0

    文章

    18

    浏览量

    14243
  • key
    key
    +关注

    关注

    0

    文章

    49

    浏览量

    12828
收藏 人收藏

    评论

    相关推荐

    接上开发板后显示EEPROM missing

    接上开发板后,设备管理其中显示:68013-EEPROMmissing在下载程序后却出现了,这样一种情况:本应该在下载程序后,68013-EEPROM missing的消息应当消失,但是在这里却没有发生上述现象
    发表于 11-21 18:22

    keil中警告missing return value

    今天遇到一个奇怪的问题:{:14:} keil编译出现DHA200A.C(820): warning C290: missing return value警告!此函数是void型main函数并无
    发表于 08-30 16:33

    仿真时,LED出现missing是怎么回事

    仿真时,LED出现missing是怎么回事
    发表于 04-17 19:16

    protues仿真出现元器件missing的问题

    protues错误改了之后出现元器件missing的问题如何解决
    发表于 07-15 20:56

    the #endif for this directive is missing 这个错误怎么解决啊?

    Error[Pe037]: the #endif for this directive is missing这个错误怎么解决啊?
    发表于 07-22 18:27

    在设计FPGA程序时,谁遇到过如下的警告啊: One or more signals are missing in the process sensitivity list.

    请问。在设计FPGA程序时,谁遇到过如下的警告啊: One or more signals are missing in the process sensitivity list.
    发表于 05-17 15:57

    mixed model MCS8051.DLL failed to authorize - Missing or invalid Customer Key.. [U1]这个啥情况求帮忙

    mixed model MCS8051.DLL failed to authorize - Missing or invalid Customer Key.. [U1]求帮忙
    发表于 01-05 12:24

    ccs6.1.2安装提示Error:missing value to go with key是为什么?

    您 好! 官网下载的最新版的离线的ccs6.1,选好安装路径点next后提示: Error:missing value to go with key,请问这是为什么? 系统环境为WIN7旗舰版64位;已关闭杀毒软件的文件系统实时防护,无法关闭其所有功能;安装路径不包含中文字符。
    发表于 05-15 09:41

    Keil编译出现“main.c(16): error C129: missing ';' before 'num'”怎么解决

    Keil编译时提示“main.c(16): error C129: missing ‘;’ before ‘num’”折腾了很久才知道是前面的“typedef unsigned char u8
    发表于 02-16 06:50

    MDK打开工程提示Missing Device是什么原因?

    MDK打开工程提示Missing Device 这个是什么原因
    发表于 08-18 08:28

    E5515C-14 Wireless Communicati

    need to be inspected for 2 missing washers on the DSP fan assembly (detailed pictures are below). If the washers are missing then t
    发表于 08-21 11:54 9次下载

    NTLDR is missing开机故障如何处理

    NTLDR is missing开机故障如何处理 故障现象:开机自检一切正常,可是在进入系统,也就是进行系统引导的时候卡住了,显示
    发表于 03-10 11:46 9287次阅读

    EFM8SB1芯科单片机调试 打开NoOverlay.hwconf文件报错

    loading the device: missing property: cslib.thresholdsettings.activesensordelta missing prop...
    发表于 11-25 19:36 9次下载
    EFM8SB1芯科单片机调试 打开NoOverlay.hwconf文件报错

    Keil编译时提示“main.c(16): error C129: missing ';' before 'num'”

    Keil编译时提示“main.c(16): error C129: missing ‘;’ before ‘num’”折腾了很久才知道是前面的“typedef unsigned char u8
    发表于 12-17 18:10 16次下载
    Keil编译时提示“main.c(16): error C129: <b class='flag-5'>missing</b> ';' before 'num'”

    Missing Container Metrics容器指标收集工具

    ./oschina_soft/missing-container-metrics.zip
    发表于 05-12 14:24 1次下载
    <b class='flag-5'>Missing</b> Container Metrics容器指标收集工具