Python描述符深入理解
Python的描述符乍眼看去简单,但是细节方面如果不注意容易掉坑,总结以下几个坑,以作备忘,先看代码:
class D:
def __get__(self, inst, owner):
if inst is None:
return self
else:
print('I am in the D.__get__')
return inst.__dict__['d'] # 返回实例的d属性
def __set__(self, inst, value):
if not isinstance(value, str):
raise AttributeError('Value must be str')
print('I am in the D.__set__')
inst.__dict__['d'] = value # 设置实例自身d属性,只能使用__dict__形式,否则又会调用__get__,从而陷入无限循环
class C:
d = D()
def __init__(self, value):
self.d = value # 在初始化的时候设置d属性,注意此时调用__get__函数,并不是设置实例本身的属性
>>> c = C('shy')
I am in the D.__set__
>>> c.__dict__
{'d': 'shy'}
>>> c.d
I am in the D.__get__
'shy'
总结:
- 描述符只能做类属性,不能作为实例属性,当一个属性是描述符时,实例查找这个属性会直接在类里面查找而忽略实例自身的空间,如上,实例自身有同名的d属性,但是当通过c.d调用的时候,调用的是描述符d。
- 在描述符里获取或者设置实例的同名属性时,需要用inst.__dict__形式访问,上面如果写成inst.d,则会陷入无限循环。
再来看一段代码:
class D:
def __get__(self, inst, owner):
if inst is None:
return self
else:
print('I am in D.__get__')
return self.value
def __set__(self, inst, value):
print('I am in D.__set__')
self.value = value
class C:
d = D()
>>> c1 = C()
>>> c2 = C()
>>> c1.d = 2
I am in D.__set__
>>> c2.d = 3
I am in D.__set__
>>> c1.d
I am in D.__get__
3
>>> c2.d
I am in D.__get__
3
class D:
def __get__(self, inst, owner):
if inst is None:
return self
else:
print('I am in D.__get__')
return inst.value
def __set__(self, inst, value):
print('I am in D.__set__')
inst.value = value
class C:
d = D()
>>> c1 = C()
>>> c2 = C()
>>> c1.d = 2
I am in D.__set__
>>> c2.d = 3
I am in D.__set__
>>> c1.d
I am in D.__get__
2
>>> c2.d
I am in D.__get__
3
总结:
属性可以保存在描述符内部,也可以保存在实例,但是如果保存为描述符内部,则为所有实例共享,所以一般把描述符的状态的信息保存在描述符内部,而把实例相关的信息保存在实例侧。
2018年12月27日
之前的一个错误,并非只要访问描述符就一定忽略其实例查找,如果只设置了__get__,优先实例查找,如果只设置了__set__函数,访问属性优先在实例字典里面查找,设置属性仍然优先类里面查找,代码如下:
# 只设置__get__,实例设置同名属性,未触发__get__
>>> class D:
def __get__(self, inst, cls):
print('I am in the D.__get__')
return inst.name
>>> class C:
name = D()
def __init__(self, value):
self.name = value
>>> c = C(2)
>>> c.name
2
#只设置__set__,实例设置同名属性,获取属性时优先实例查找,设置属性时优先类查找
>>> class D:
def __set__(self, inst, value):
print('I am in D __set__')
inst.__dict__['name'] = value
>>> class C:
name = D()
>>> c = C()
>>> c.name = 2
I am in D __set__
>>> c.__dict__
{'name': 2}
>>> c.name
2
>>> c.name = 3
I am in D __set__
Python描述符深入理解的更多相关文章
- 杂项之python描述符协议
杂项之python描述符协议 本节内容 由来 描述符协议概念 类的静态方法及类方法实现原理 类作为装饰器使用 1. 由来 闲来无事去看了看django中的内置分页方法,发现里面用到了类作为装饰器来使用 ...
- python描述符(descriptor)、属性(property)、函数(类)装饰器(decorator )原理实例详解
1.前言 Python的描述符是接触到Python核心编程中一个比较难以理解的内容,自己在学习的过程中也遇到过很多的疑惑,通过google和阅读源码,现将自己的理解和心得记录下来,也为正在为了该问题 ...
- 【转载】Python 描述符简介
来源:Alex Starostin 链接:www.ibm.com/developerworks/cn/opensource/os-pythondescriptors/ 关于Python@修饰符的文章可 ...
- Python描述符 (descriptor) 详解
1.什么是描述符? python描述符是一个“绑定行为”的对象属性,在描述符协议中,它可以通过方法重写属性的访问.这些方法有 __get__(), __set__(), 和__delete__().如 ...
- python描述符学习
目录 一.对象属性的访问控制 二.描述符基本理解 三.基本使用 四.使用描述符完成property.classmethod.staticmethod自定义实现 1.property的自定义实现 2.c ...
- python描述符descriptor(一)
Python 描述符是一种创建托管属性的方法.每当一个属性被查询时,一个动作就会发生.这个动作默认是get,set或者delete.不过,有时候某个应用可能会有 更多的需求,需要你设计一些更复杂的动作 ...
- python描述符 descriptor
descriptor 在python中,如果一个新式类定义了__get__, __set__, __delete__方法中的一个或者多个,那么称之为descriptor.descriptor通常用来改 ...
- USB学习小记-HID类键盘的报告描述符的理解
前言 断断续续的学习了将近三个月,才把USB的HID类搞明白,速度真是够慢的.利用晚上+周末的时间学习自己的东西确实是必要的,不过效率是有点低,以后要更专注一些才行,希望自己能做到吧. 在学习过程中, ...
- Python描述符的使用
Python描述符的使用 前言 作为一位python的使用者,你可能使用python有一段时间了,但是对于python中的描述符却未必使用过,接下来是对描述符使用的介绍 场景介绍 为了引入描述符的使用 ...
随机推荐
- node.js开发指南读书笔记(1)
3.1 开始使用Node.js编程 3.1.1 Hello World 将以下源代码保存到helloworld.js文件中 console.log('Hello World!'); console.l ...
- c++之继承与派生
再来回顾下继承派生的语法. 继承方式显示有三种(public, protected, privatez),隐式默认private.所谓继承方式,是指派生类对基类成员的访问权限控制. 派生类构造函数定义 ...
- 【HDU5861】Road
题意 有n个村庄排成一排,有n-1条路将他们连在一起.每条路开放一天都会花费一定数量的钱.你可以选择打开或者关上任意条路在任意一天,但是每条路只能打开和关闭一次.我们知道m天的运输计划.每天都有一辆马 ...
- 算法描述》关于LIS的nlogn方法
上次TYVJ有一道裸LIS,然而我当时直接打了一个N^2暴力就草草了事,然后就ZZ了,只拿了60分,其实NlogN的LIS和N^2的差的不多,只是没有N^2,好想罢了,鉴于某学弟的要求,所以就重现一下 ...
- PHP内核介绍及扩展开发指南—Extensions 的编写
Extensions 的编写 理解了这些运行机制以后,本章着手介绍Extensions 的编写,但凡写程序的人都知道hello world,那好,就从hello world开始. 1.1Hello W ...
- inux php pdo mysql 扩展
今天在本机部署了一个pdo项目,发现一些问题,真没想到pdo mysql,不容易装啊,哈哈,我说的不容易,是因为php5.3以前版本,yum源里面根本没有.部署后就报,Undefined class ...
- 修改Tomcat可支持get传参方式的url长度,get形式
maxHttpHeaderSize="8192"加在 <Connector port="8081" maxHttpHeaderSize="314 ...
- centos 搭建docker环境
我有一台便宜的腾讯云服务器,当然配置自然也是最低的,只是用来平常玩一玩,学习的用处,下面介绍一下我在上面搭建docker的心得,共勉一下. 安装与配置 Docker 安装 Docker Docker ...
- 9-python 的ProxyHandler处理器(代理设置)
ProxyHandler处理器(代理设置) 使用代理IP,这是爬虫/反爬虫的第二大招,通常也是最好用的. 很多网站会检测某一段时间某个IP的访问次数(通过流量统计,系统日志等),如果访问次数多的不像正 ...
- How to install Freemind 1.0.1 to Ubuntu 14
安装了Freemind0.9后发现不能打开windows的1.0.1保存的*.mm文件,便对版本开始升级. 1. 从http://freemind.sourceforge.net/wiki/index ...