轻松理解Python 中的 descriptor


Posted in Python onSeptember 15, 2017

定义

通常,一个 descriptor 是具有“绑定行为”的对象属性。所绑定行为可通过 descriptor 协议被自定义的 __get__() , __set__() 和 __delete__() 方法重写。如果一个对象的上述三个方法任意一个被重写,则就可被称为 descriptor。

属性的默认操作是从对象字典中获取、设置和删除一个属性。例如,a.x 有一个查找链,先 a.__dict__['x'] ,若没有则 type(a).__dict__['x'] ,若没有增往上查找父类直到元类。如果查找链中,对象被定义了 descriptor 方法,Python 就会覆盖默认行为。

Descriptor 是一个强大的工具,虽然开发者不常接触到它,但它其实就是类、属性、函数、方法、静态方法、类方法以及 super() 背后的运行机制。

Descriptor 协议

三个方法原型如下所示:

descr.__get__(self, obj, type=None) --> value
descr.__set__(self, obj, value) --> None
descr.__delete__(self, obj) --> None

数据 descriptor 是同时具有 __get__() 和 __set__() 方法的对象,若只有 __get__() 方法,则为非数据 descriptor。如果实例字典中有和数据 descriptor 同名的入口,则数据 descriptor 优先级更高。相反,非数据 descriptor 优先级低。

让 __set__() 方法抛出异常,就能创建一个只读数据 descriptor。

调用 descriptor

descriptor 可以直接通过方法名调用。例如, d.__get__(obj) 。

而通过访问对象属性,自动调用 descriptor 才是更通用的做法。例如,如果 d 定义了方法 __get__() ,则 obj.d 会调用 d.__get__(obj) 。

对于对象, b.x 会被转换成 type(b).__dict__['x'].__get__(b, type(b)) 。而对于类(是的,类也可以调用), B.x 会被转换成 B.__dict__['x'].__get__(None, B) 。

Descriptor 例子

class RevealAccess(object):
  """A data descriptor that sets and returns values
    normally and prints a message logging their access.
  """
  def __init__(self, initval=None, name='var'):
    self.val = initval
    self.name = name
  def __get__(self, obj, objtype):
    print('Retrieving', self.name)
    return self.val
  def __set__(self, obj, val):
    print('Updating', self.name)
    self.val = val
>>> class MyClass(object):
...   x = RevealAccess(10, 'var "x"')
...   y = 5
...
>>> m = MyClass()
>>> m.x
Retrieving var "x"
10
>>> m.x = 20
Updating var "x"
>>> m.x
Retrieving var "x"
20
>>> m.y
5

总结

以上所述是小编给大家介绍的Python 中的 descriptor,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Python 相关文章推荐
Python中文编码那些事
Jun 25 Python
用Python编写一个简单的FUSE文件系统的教程
Apr 02 Python
用Python进行行为驱动开发的入门教程
Apr 23 Python
对dataframe进行列相加,行相加的实例
Jun 08 Python
Python第三方Window模块文件的几种安装方法
Nov 22 Python
python学生管理系统开发
Jan 30 Python
kali中python版本的切换方法
Jul 11 Python
python groupby 函数 as_index详解
Dec 16 Python
用python打开摄像头并把图像传回qq邮箱(Pyinstaller打包)
May 17 Python
查找适用于matplotlib的中文字体名称与实际文件名对应关系的方法
Jan 05 Python
pytest fixtures装饰器的使用和如何控制用例的执行顺序
Jan 28 Python
python单向链表实例详解
May 25 Python
Python 实现购物商城,含有用户入口和商家入口的示例
Sep 15 #Python
python中reload(module)的用法示例详解
Sep 15 #Python
Python 关于反射和类的特殊成员方法
Sep 14 #Python
在Python中执行系统命令的方法示例详解
Sep 14 #Python
关于Python如何避免循环导入问题详解
Sep 14 #Python
Python实现随机选择元素功能
Sep 14 #Python
python自动化脚本安装指定版本python环境详解
Sep 14 #Python
You might like
利用discuz自带通行证整合dedecms的方法以及文件下载
2007/03/06 PHP
说明的比较细的php 正则学习实例
2008/07/30 PHP
浅谈PHP正则表达式中修饰符/i, /is, /s, /isU
2014/10/21 PHP
PHP实现的基于单向链表解决约瑟夫环问题示例
2017/09/30 PHP
安装PHP扩展时解压官方 tgz 文件后没有configure文件无法进行配置编译的问题
2020/08/26 PHP
IE与Firefox在JavaScript上的7个不同写法小结
2009/09/14 Javascript
读jQuery之八 包装事件对象
2011/06/21 Javascript
jquery 图片缩放拖动的简单实例
2014/01/08 Javascript
Jquery遍历checkbox获取选中项value值的方法
2014/02/13 Javascript
jQuery中prepend()方法使用详解
2015/08/11 Javascript
jQuery+PHP实现可编辑表格字段内容并实时保存
2015/10/09 Javascript
js跨域资源共享 基础篇
2016/07/02 Javascript
浅谈jQuery中的checkbox问题
2016/08/10 Javascript
jQuery实现ToolTip元素定位显示功能示例
2016/11/23 Javascript
BootStrap Table前台和后台分页对JSON格式的要求
2017/06/28 Javascript
vue.js select下拉框绑定和取值方法
2018/03/03 Javascript
详解nodejs 配置文件处理方案
2019/01/02 NodeJs
jQuery实现提交表单时不提交隐藏div中input的方法
2019/10/08 jQuery
通过实例解析JavaScript常用排序算法
2020/09/02 Javascript
Nuxt 嵌套路由nuxt-child组件用法(父子页面组件的传值)
2020/11/05 Javascript
微信小程序自定义tabBar的踩坑实践记录
2020/11/06 Javascript
Python urllib模块urlopen()与urlretrieve()详解
2013/11/01 Python
Python socket C/S结构的聊天室应用实现
2014/11/30 Python
Python日志模块logging简介
2015/04/13 Python
python批量修改文件编码格式的方法
2018/05/31 Python
在macOS上搭建python环境的实现方法
2019/08/13 Python
Python3安装pip工具的详细步骤
2019/10/14 Python
Python内置方法实现字符串的秘钥加解密(推荐)
2019/12/09 Python
利用python+request通过接口实现人员通行记录上传功能
2021/01/13 Python
轻金属冶金专业毕业生自荐信
2013/11/02 职场文书
致跳远、跳高运动员广播稿
2014/01/09 职场文书
安徽导游词
2015/02/12 职场文书
物流业务员岗位职责
2015/04/03 职场文书
基于tensorflow权重文件的解读
2021/05/26 Python
详解Flutter和Dart取消Future的三种方法
2022/04/07 Java/Android
Python Matplotlib绘制动画的代码详解
2022/05/30 Python