这篇文章主要介绍“Python中missing()函数怎么使用”,在日常操作中,相信很多人在Python中missing()函数怎么使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python中missing()函数怎么使用”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
本文主要关注的问题有:
(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 中分明包含这个属性:
出处:3. Data model — Python 3.10.1 documentationmissing#object.missing
也就是说,理论上 object 类中会预定义missing,其文档证明了这一点,然而实际上它并没有被定义!文档与现实出现了偏差!
如此一来,当 dict 的子类(例如 defaultdict 和 Counter)在定义missing 时,这个魔术方法事实上只属于该子类,也就是说,