轻松理解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的词法分析与语法分析
May 18 Python
Python爬虫框架Scrapy安装使用步骤
Apr 01 Python
python中requests小技巧
May 10 Python
apache部署python程序出现503错误的解决方法
Jul 24 Python
python3使用smtplib实现发送邮件功能
May 22 Python
Python将json文件写入ES数据库的方法
Apr 10 Python
Python3 itchat实现微信定时发送群消息的实例代码
Jul 12 Python
Django打印出在数据库中执行的语句问题
Jul 25 Python
基于python plotly交互式图表大全
Dec 07 Python
python设置环境变量的作用整理
Feb 17 Python
Python流程控制语句的深入讲解
Jun 15 Python
python 基于PYMYSQL使用MYSQL数据库
Dec 24 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
用js进行url编码后用php反解以及用php实现js的escape功能函数总结
2010/02/08 PHP
PHP网站备份程序代码分享
2011/06/10 PHP
ThinkPHP中的三大自动简介
2014/08/22 PHP
PHP下的Oracle客户端扩展(OCI8)安装教程
2014/09/10 PHP
深入探究PHP的多进程编程方法
2015/08/18 PHP
php使用ftp实现文件上传与下载功能
2017/07/21 PHP
js实现权限树的更新权限时的全选全消功能
2009/02/17 Javascript
JavaScript 获取事件对象的注意点
2009/07/29 Javascript
Js控制弹窗实现在任意分辨率下居中显示
2013/08/01 Javascript
nodeType属性返回被选节点的节点类型介绍
2013/11/22 Javascript
jQuery 删除/替换DOM元素的几种方式
2014/05/20 Javascript
基于angularjs实现图片放大镜效果
2016/08/31 Javascript
fetch 使用及如何接收JS传值
2017/11/11 Javascript
ES6/JavaScript使用技巧分享
2017/12/14 Javascript
vue项目中应用ueditor自定义上传按钮功能
2018/04/27 Javascript
ajax与jsonp的区别及用法
2018/10/16 Javascript
详解React中合并单元格的正确写法
2019/01/08 Javascript
vue实现瀑布流组件滑动加载更多
2020/03/10 Javascript
Vue 电商后台管理项目阶段性总结(推荐)
2020/08/22 Javascript
[00:36]我的中国心——Serenity vs Fnatic
2018/08/21 DOTA
python实现去除下载电影和电视剧文件名中的多余字符的方法
2014/09/23 Python
Python标准库笔记struct模块的使用
2018/02/22 Python
Python线性拟合实现函数与用法示例
2018/12/13 Python
Django静态资源部署404问题解决方案
2020/05/11 Python
在django admin中配置搜索域是一个外键时的处理方法
2020/05/20 Python
推荐技术人员一款Python开源库(造数据神器)
2020/07/08 Python
基于CSS3的animation属性实现微信拍一拍动画效果
2020/06/22 HTML / CSS
澳大利亚先进的皮肤和激光诊所购物网站:Soho Skincare
2018/10/15 全球购物
文体活动实施方案
2014/03/27 职场文书
竞选团支书演讲稿
2014/04/28 职场文书
专家推荐信范文
2015/03/26 职场文书
小学教师党员承诺书
2015/04/27 职场文书
餐厅开业活动方案
2019/07/08 职场文书
迎客户欢迎词三篇
2019/09/27 职场文书
css3属性选择器 “~”(波浪号) “,”(逗号) “+”(加号)和 “>”(大于号)
2022/04/19 HTML / CSS
详解Spring Security如何在权限中使用通配符
2022/06/28 Java/Android