• Python的descriptor (2)


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

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

    Python代码  收藏代码
    1. >>> def hello():  
    2.     pass  
    3.   
    4. >>> dir(hello)  
    5. ['__call__', '__class__', '__delattr__', '__dict__', '__doc__', '<span style="color: #ff0000;">__get__</span>  
    6.   
    7.   
    8.   
    9. ', '__getattribute__',   
    10. '__hash__', '__init__', '__module__', '__name__', '__new__',   
    11. '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__str__', 'func_closure',   
    12. 'func_code', 'func_defaults', 'func_dict', 'func_doc', 'func_globals', 'func_name']  
    13. >>>   

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

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

    Python代码  收藏代码
    1. >>> class T(object):  
    2.     def hello(self):  
    3.         pass  
    4.   
    5. >>> T.__dict__['hello']  
    6. <function hello at 0x00CD7EB0>  
    7. >>>   

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

    >>> T.hello   获取T的hello属性,根据查找策略,从T的__dict__中找到了,找到的是<function hello at 0x00CD7EB0>,但不会直接返回<function hello at 0x00CD7EB0>,因为它有__get__方法,所以返回的是调用它的__get__(None, T)的结果:一个unbound方法。
    
    
    <unbound method T.hello>
    >>> f = T.__dict__['hello']   #直接从T的__dict__中获取hello,不会执行查找策略,直接返回了<function hello at 0x00CD7EB0>
    
    
    >>> f
    <function hello at 0x00CD7EB0>
    >>> t = T()                 
    >>> t.hello                     #从实例获取属性,返回的是调用<function hello at 0x00CD7EB0>的__get__(t, T)的结果:一个bound方法。
    
    
    
    <bound method T.hello of <__main__.T object at 0x00CDAD10>>
    >>> 

     为了证实我们上面的说法,在继续下面的代码(f还是上面的<function hello at 0x00CD7EB0>):

    Python代码  收藏代码
    1. >>> f.__get__(None, T)  
    2. <unbound method T.hello>  
    3. >>> f.__get__(t, T)  
    4. <bound method T.hello of <__main__.T object at 0x00CDAD10>>  

     好极了!

    总结一下:

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

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

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

    Python代码  收藏代码
    1. >>> type(f)  
    2. <type 'function'>  
    3. >>> type(t.hello)  
    4. <type 'instancemethod'>  
    5. >>> type(T.hello)  
    6. <type 'instancemethod'>  
    7. >>>   

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

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

    Python代码  收藏代码
    1. >>> dir(t.hello)  
    2. ['__call__', '__class__', '__cmp__', '__delattr__', '__doc__', '__get__', '__getattribute__',   
    3. '__hash__', '__init__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__',   
    4. '__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能将一个函数转换成类方法,类方法的第一个隐含参数是类本身 (普通方法的第一个隐含参数是实例本身),类方法即可从类调用,也可以从实例调用(普通方法只能从实例调用)。

    Python代码  收藏代码
    1. >>> class T(object):  
    2.     def hello(cls):  
    3.         print 'hello', cls  
    4.     hello = classmethod(hello)   #两个作用:把hello装换成类方法,同时隐藏作为普通方法的hello  
    5.   
    6.       
    7. >>> t = T()  
    8. >>> t.hello()  
    9. hello <class '__main__.T'>  
    10. >>> T.hello()  
    11. hello <class '__main__.T'>  
    12. >>>   

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

    Python代码  收藏代码
    1. >>> t.hello  
    2. <bound method type.hello of <class '__main__.T'>>  
    3. >>> type(t.hello)  
    4. <type 'instancemethod'>  
    5. >>> T.hello  
    6. <bound method type.hello of <class '__main__.T'>>  
    7. >>> type(T.hello)  
    8. <type 'instancemethod'>  
    9. >>>   

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

    Python代码  收藏代码
    1. >>> t.hello.im_self  
    2. <class '__main__.T'>  
    3. >>> t.hello.im_class  
    4. <type 'type'>  
    5. >>> t.hello.im_func  
    6. <function hello at 0x011A40B0>  
    7. >>>   

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

    staticmethod

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

    Python代码  收藏代码
    1. class T(object):  
    2.     def hello():  
    3.         print 'hello'  
    4.     hello = staticmethod(hello)  
    5.   
    6.       
    7. >>> T.hello()   #没有隐含的第一个参数  
    8. hello  
    9. >>> T.hello  
    10. <function hello at 0x011A4270>  
    11. >>>   

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

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

  • 相关阅读:
    锐浪报表(Grid++Report)问题-子报表获取不到父报表参数问题解决
    C#PictureBox实现一直显示提示信息
    C#.Net ComboBox控件设置DropDownList之后背景颜色问题,以及发现的微软的一个BUG
    Visual Studio 2019移除/禁用Live Share按钮
    robotframework导入selenium2library变红
    robotframework selenium2library中关键字缺失的问题
    Jmeter之JSON提取器的使用
    Jmeter中unicode转码问题处理
    Jmeter之关联接口处理
    jmeter之用户自定义的变量
  • 原文地址:https://www.cnblogs.com/rrxc/p/4269182.html
Copyright © 2020-2023  润新知