Python描述器descriptor详解
admin
2023-07-31 02:09:50
0

前面说了descriptor,这个东西其实和Java的setter,getter有点像。但这个descriptor和上文中我们开始提到的函数方法这些东西有什么关系呢?

所有的函数都可以是descriptor,因为它有__get__方法。

复制代码 代码如下:
>>> def hello(): 
    pass 
>>> dir(hello) 
[\’__call__\’, \’__class__\’, \’__delattr__\’, \’__dict__\’, \’__doc__\’, \'__get__ 
\’, \’__getattribute__\’,  
\’__hash__\’, \’__init__\’, \’__module__\’, \’__name__\’, \’__new__\’,  
\’__reduce__\’, \’__reduce_ex__\’, \’__repr__\’, \’__setattr__\’, \’__str__\’, \’func_closure\’,  
\’func_code\’, \’func_defaults\’, \’func_dict\’, \’func_doc\’, \’func_globals\’, \’func_name\’] 
>>>  

 注意,函数对象没有__set__和__del__方法,所以它是个non-data descriptor.

方法其实也是函数,如下:

复制代码 代码如下:
>>> class T(object): 
    def hello(self): 
        pass 
>>> T.__dict__[\’hello\’] 
 
>>> 

 或者,我们可以把方法看成特殊的函数,只是它们存在于类 中,获取函数属性时,返回的不是函数本身(比如上面的),而是返回函数的__get__方法的返回值,接着上面类T的定义:

>>> T.hello   获取T的hello属性,根据查找策略,从T的__dict__中找到了,找到的是,但不会直接返回,因为它有__get__方法,所以返回的是调用它的__get__(None, T)的结果:一个unbound方法。


>>> f = T.__dict__[\’hello\’]   #直接从T的__dict__中获取hello,不会执行查找策略,直接返回了

复制代码 代码如下:
>>> f

>>> t = T()                
>>> t.hello                     #从实例获取属性,返回的是调用的__get__(t, T)的结果:一个bound方法。

复制代码 代码如下:
>
>>>

 为了证实我们上面的说法,在继续下面的代码(f还是上面的):

复制代码 代码如下:
>>> f.__get__(None, T) 
 
>>> f.__get__(t, T) 

 好极了!

总结一下:

      1.所有的函数都有__get__方法

      2.当函数位于类的__dict__中时,这个函数可以认为是个方法,通过类或实例获取该函数时,返回的不是函数本身,而是它的__get__方法返回值。

我承认我可能误导你认为方法就是函数,是特殊的函数。其实方法和函数还是有区别的,准确的说:方法就是方法,函数就是函数。

复制代码 代码如下:
>>> type(f) 
 
>>> type(t.hello) 
 
>>> type(T.hello) 
 
>>>  

 函数是function类型的,method是instancemethod(这是普通的实例方法,后面会提到classmethod和staticmethod)。

关于unbound method和bound method,再多说两句。在c实现中,它们是同一个对象(它们都是instancemethod类型的),我们先看看它们里面到底是什么

复制代码 代码如下:
>>> dir(t.hello) 
[\’__call__\’, \’__class__\’, \’__cmp__\’, \’__delattr__\’, \’__doc__\’, \’__get__\’, \’__getattribute__\’,  
\’__hash__\’, \’__init__\’, \’__new__\’, \’__reduce__\’, \’__reduce_ex__\’, \’__repr__\’, \’__setattr__\’,  
\’__str__\’, \’im_class\’, \’im_func\’, \’im_self\’] 

 __call__说明它们是个可调用对象,而且我们还可以猜测,这个__call__的实现应该大致是:转调另外一个函数(我们期望的哪个,比如上面的hello),并以对象作为第一参数。

要 注意的是im_class,im_func,im_self。这几个东西我们并不陌生,在t.hello里,它们分别代表T,hello(这里是存储在 T.__dict__里的函数hello)和t。有了这些我们可以大致想象如何纯Python实现一个instancemethod了:)。

其实还有几个内建函数都和descriptor有关,下面简单说说。

classmethod

classmethod能将一个函数转换成类方法,类方法的第一个隐含参数是类本身 (普通方法的第一个隐含参数是实例本身),类方法即可从类调用,也可以从实例调用(普通方法只能从实例调用)。

复制代码 代码如下:
>>> class T(object): 
    def hello(cls): 
        print \’hello\’, cls 
    hello = classmethod(hello)   #两个作用:把hello装换成类方法,同时隐藏作为普通方法的hello  
>>> t = T() 
>>> t.hello() 
hello  
>>> T.hello() 
hello  
>>>  

 注意:classmethod是个类,不是函数。classmethod类有__get__方法,所以,上面的t.hello和T.hello获得实际上是classmethod的__get__方法返回值

复制代码 代码如下:
>>> t.hello 

>>> type(t.hello) 
 
>>> T.hello 

>>> type(T.hello) 
 
>>>  

 从 上面可以看出,t.hello和T.hello是instancemethod类型的,而且是绑定在T上的。也就是说classmethod的 __get__方法返回了一个instancemethod对象。从前面对instancemethod的分析上,我们应该可以推断:t.hello的 im_self是T,im_class是type(T是type的实例),im_func是函数hello

复制代码 代码如下:
>>> t.hello.im_self 
 
>>> t.hello.im_class 
 
>>> t.hello.im_func 
 
>>>  

 完全一致!所以实现一个纯Python的classmethod也不难:)

staticmethod

staticmethod能将一个函数转换成静态方法,静态方法没有隐含的第一个参数。

复制代码 代码如下:
class T(object): 
    def hello(): 
        print \’hello\’ 
    hello = staticmethod(hello)     
>>> T.hello()   #没有隐含的第一个参数 
hello 
>>> T.hello 
 
>>> 

 T.hello直接返回了一个函数。猜想staticmethod类的__get__方法应该是直接返回了对象本身。

还有一个property,和上面两个差不多,它是个data descriptor。

相关内容

热门资讯

500 行 Python 代码... 语法分析器描述了一个句子的语法结构,用来帮助其他的应用进行推理。自然语言引入了很多意外的歧义,以我们...
定时清理删除C:\Progra... C:\Program Files (x86)下面很多scoped_dir开头的文件夹 写个批处理 定...
65536是2的几次方 计算2... 65536是2的16次方:65536=2⁶ 65536是256的2次方:65536=256 6553...
Mobi、epub格式电子书如... 在wps里全局设置里有一个文件关联,打开,勾选电子书文件选项就可以了。
scoped_dir32_70... 一台虚拟机C盘总是莫名奇妙的空间用完,导致很多软件没法再运行。经过仔细检查发现是C:\Program...
小程序支付时提示:appid和... [Q]小程序支付时提示:appid和mch_id不匹配 [A]小程序和微信支付没有进行关联,访问“小...
pycparser 是一个用... `pycparser` 是一个用 Python 编写的 C 语言解析器。它可以用来解析 C 代码并构...
微信小程序使用slider实现... 众所周知哈,微信小程序里面的音频播放是没有进度条的,但最近有个项目呢,客户要求音频要有进度条控制,所...
python查找阿姆斯特朗数 题目解释 如果一个n位正整数等于其各位数字的n次方之和,则称该数为阿姆斯特朗数。 例如1^3 + 5...
Apache Doris 2.... 亲爱的社区小伙伴们,我们很高兴地向大家宣布,Apache Doris 2.0.0 版本已于...