Python描述器descriptor详解


Posted in Python onFebruary 03, 2015

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

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

>>> def hello():  

    pass  

>>> dir(hello)  

['__call__', '__class__', '__delattr__', '__dict__', '__doc__', '<span style="color: #ff0000;">__get__</span>  

', '__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']  

<function hello at 0x00CD7EB0>  

>>> 

 或者,我们可以把方法看成特殊的函数,只是它们存在于类 中,获取函数属性时,返回的不是函数本身(比如上面的<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>):

>>> f.__get__(None, T)  

<unbound method T.hello>  

>>> f.__get__(t, T)  

<bound method T.hello of <__main__.T object at 0x00CDAD10>> 

 好极了!

总结一下:

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

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

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

>>> type(f)  

<type 'function'>  

>>> type(t.hello)  

<type 'instancemethod'>  

>>> type(T.hello)  

<type 'instancemethod'>  

>>>  

 函数是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 <class '__main__.T'>  

>>> T.hello()  

hello <class '__main__.T'>  

>>>  

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

>>> t.hello  

<bound method type.hello of <class '__main__.T'>>  

>>> type(t.hello)  

<type 'instancemethod'>  

>>> T.hello  

<bound method type.hello of <class '__main__.T'>>  

>>> type(T.hello)  

<type 'instancemethod'>  

>>>  

 从 上面可以看出,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  

<class '__main__.T'>  

>>> t.hello.im_class  

<type 'type'>  

>>> t.hello.im_func  

<function hello at 0x011A40B0>  

>>>  

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

staticmethod

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

class T(object):  

    def hello():  

        print 'hello'  

    hello = staticmethod(hello)      

>>> T.hello()   #没有隐含的第一个参数  

hello  

>>> T.hello  

<function hello at 0x011A4270>  

>>> 

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

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

Python 相关文章推荐
python实现探测socket和web服务示例
Mar 28 Python
python实现绘制树枝简单示例
Jul 24 Python
零基础写python爬虫之爬虫编写全记录
Nov 06 Python
在Python中使用SimpleParse模块进行解析的教程
Apr 11 Python
Python搭建HTTP服务器和FTP服务器
Mar 09 Python
python 时间信息“2018-02-04 18:23:35“ 解析成字典形式的结果代码详解
Apr 19 Python
python语音识别实践之百度语音API
Aug 30 Python
PyTorch 对应点相乘、矩阵相乘实例
Dec 27 Python
python:批量统计xml中各类目标的数量案例
Mar 10 Python
Python3实现个位数字和十位数字对调, 其乘积不变
May 03 Python
使用anaconda安装pytorch的实现步骤
Sep 03 Python
python实现简单的名片管理系统
Apr 26 Python
理解Python中的With语句
Feb 02 #Python
Linux环境下MySQL-python安装过程分享
Feb 02 #Python
Python中用pycurl监控http响应时间脚本分享
Feb 02 #Python
Python列表(list)常用操作方法小结
Feb 02 #Python
Python Sleep休眠函数使用简单实例
Feb 02 #Python
Python中实现从目录中过滤出指定文件类型的文件
Feb 02 #Python
Python实现二分法算法实例
Feb 02 #Python
You might like
phpMyAdmin下载、安装和使用入门教程
2007/05/31 PHP
PHP根据session与cookie用户登录状态操作类的代码
2016/05/13 PHP
一个简单安全的PHP验证码类、PHP验证码
2016/09/24 PHP
Windows服务器中PHP如何安装redis扩展
2019/09/27 PHP
JavaScript高级程序设计(第3版)学习笔记8 js函数(中)
2012/10/11 Javascript
Web Inspector:关于在 Sublime Text 中调试Js的介绍
2013/04/18 Javascript
JQuery触发事件例如click
2013/09/11 Javascript
JavaScript的常见兼容问题及相关解决方法(chrome/IE/firefox)
2013/12/31 Javascript
Jquery实现简单的轮播效果(代码管用)
2016/03/14 Javascript
JavaScript之underscore_动力节点Java学院整理
2017/07/03 Javascript
基于JavaScript+HTML5 实现打地鼠小游戏逻辑流程图文详解(附完整代码)
2017/11/02 Javascript
vue组件name的作用小结
2018/05/23 Javascript
详解JS函数stack size计算方法
2018/06/18 Javascript
vue.js2.0 实现better-scroll的滚动效果实例详解
2018/08/13 Javascript
详解angular应用容器化部署
2018/08/14 Javascript
Vue.js中 v-model 指令的修饰符详解
2018/12/03 Javascript
ES6 如何改变JS内置行为的代理与反射
2019/02/11 Javascript
微信小程序自定义扫码功能界面的实现代码
2020/07/02 Javascript
swiperjs实现导航与tab页的联动
2020/12/13 Javascript
Python运行报错UnicodeDecodeError的解决方法
2016/06/07 Python
Python cookbook(数据结构与算法)字典相关计算问题示例
2018/02/18 Python
解决pyttsx3无法封装的问题
2018/12/24 Python
python实现贪吃蛇游戏
2020/03/21 Python
基于python2.7实现图形密码生成器的实例代码
2019/11/05 Python
python随机数分布random均匀分布实例
2019/11/27 Python
python except异常处理之后不退出,解决异常继续执行的实现
2020/04/25 Python
Python pip安装第三方库实现过程解析
2020/07/09 Python
Python 实现PS滤镜中的径向模糊特效
2020/12/03 Python
Html5实现如何在两个div元素之间拖放图像
2013/03/29 HTML / CSS
英语教师自荐信
2014/05/26 职场文书
主要负责人任命书
2014/06/06 职场文书
4s店活动策划方案
2014/08/25 职场文书
办理房产过户的委托书
2014/09/14 职场文书
一般基层干部群众路线教育实践活动个人对照检查材料
2014/11/04 职场文书
2015年财务个人工作总结范文
2015/05/22 职场文书
关于环保的宣传稿
2015/07/23 职场文书