Python黑魔法之描述符
时间:2025-11-04 17:19:17 出处:应用开发阅读(143)
引言
Descriptors(描述符)是黑魔Python语言中一个深奥但很重要的一个黑魔法,它被广泛应用于Python语言的法之符内核,熟练掌握描述符将会为Python程序员的描述工具箱添加一个额外的技巧。本文我将讲述描述符的黑魔定义以及一些常见的场景,并且在文末会补充一下__getattr,法之符__getattribute__,描述 __getitem__这三个同样涉及到属性访问的魔术方法。

描述符的黑魔定义
descr__get__(self, obj, objtype=None) --> value descr.__set__(self, obj, value) --> None descr.__delete__(self, obj) --> None只要一个object attribute(对象属性)定义了上面三个方法中的任意一个,那么这个类就可以被称为描述符类。法之符
描述符基础
下面这个例子中我们创建了一个RevealAcess类,描述并且实现了__get__方法,黑魔现在这个类可以被称为一个描述符类。法之符
class RevealAccess(object): def __get__(self,描述 obj, objtype): print(self in RevealAccess: {}.format(self)) print(self: {}\nobj: {}\nobjtype: {}.format(self, obj, objtype)) class MyClass(object): x = RevealAccess() def test(self): print(self in MyClass: {}.format(self))EX1实例属性
接下来我们来看一下__get__方法的各个参数的含义,在下面这个例子中,黑魔self即RevealAccess类的法之符实例x,obj即MyClass类的描述实例m,objtype顾名思义就是MyClass类自身。从输出语句可以看出,m.x访问描述符x会调用__get__方法。
>>> m = MyClass() >>> m.test() self in MyClass: <__main__.MyClass object at 0x7f19d4e42160> >>> m.x self in RevealAccess: <__main__.RevealAccess object at 0x7f19d4e420f0> self: <__main__.RevealAccess object at 0x7f19d4e420f0> obj: <__main__.MyClass object at 0x7f19d4e42160> objtype: <class __main__.MyClass>EX2类属性
如果通过类直接访问属性x,云服务器提供商那么obj接直接为None,这还是比较好理解,因为不存在MyClass的实例。
>>> MyClass.x self in RevealAccess: <__main__.RevealAccess object at 0x7f53651070f0> self: <__main__.RevealAccess object at 0x7f53651070f0> obj: None objtype: <class __main__.MyClass>描述符的原理
描述符触发
上面这个例子中,我们分别从实例属性和类属性的角度列举了描述符的用法,下面我们来仔细分析一下内部的原理:
如果是对实例属性进行访问,实际上调用了基类object的__getattribute__方法,在这个方法中将obj.d转译成了type(obj).__dict__[d].__get__(obj, type(obj))。 如果是对类属性进行访问,相当于调用了元类type的__getattribute__方法,它将cls.d转译成cls.__dict__[d].__get__(None, cls),这里__get__()的obj为的None,因为不存在实例。简单讲一下__getattribute__魔术方法,这个方法在我们访问一个对象的属性的时候会被无条件调用,详细的细节比如和__getattr, __getitem__的区别我会在文章的末尾做一个额外的WordPress模板补充,我们暂时并不深究。
描述符优先级
首先,描述符分为两种:
如果一个对象同时定义了__get__()和__set__()方法,则这个描述符被称为data descriptor。 如果一个对象只定义了__get__()方法,则这个描述符被称为non-data descriptor。我们对属性进行访问的时候存在下面四种情况:
data descriptor instance dict non-data descriptor __getattr__()它们的优先级大小是:
data descriptor > instance dict > non-data descriptor > __getattr__()这是什么意思呢?就是说如果实例对象obj中出现了同名的data descriptor->d 和 instance attribute->d,obj.d对属性d进行访问的时候,由于data descriptor具有更高的优先级,Python便会调用type(obj).__dict__[d].__get__(obj, type(obj))而不是调用obj.__dict__[‘d’]。但是如果描述符是个non-data descriptor,Python则会调用obj.__dict__[d]。
Property
每次使用描述符的时候都定义一个描述符类,这样看起来非常繁琐。Python提供了一种简洁的方式用来向属性添加数据描述符。
property(fget=None, fset=None, fdel=None, doc=None) -> property attributefget、fset和fdel分别是类的getter、亿华云setter和deleter方法。我们通过下面的一个示例来说明如何使用Property:
class Account(object): def __init__(self): self._acct_num = None def get_acct_num(self): return self._acct_num def set_acct_num(self, value): self._acct_num = value def del_acct_num(self): del self._acct_num acct_num = property(get_acct_num, set_acct_num, del_acct_num, _acct_num property.)如果acct是Account的一个实例,acct.acct_num将会调用getter,acct.acct_num = value将调用setter,del acct_num.acct_num将调用deleter。
>>> acct = Account() >>> acct.acct_num = 1000 >>> acct.acct_num 1000Python也提供了@property装饰器,对于简单的应用场景可以使用它来创建属性。一个属性对象拥有getter,setter和deleter装饰器方法,可以使用它们通过对应的被装饰函数的accessor函数创建属性的拷贝。
class Account(object): def __init__(self): self._acct_num = None @property # the _acct_num property. the decorator creates a read-only property def acct_num(self): return self._acct_num @acct_num.setter # the _acct_num property setter makes the property writeable def set_acct_num(self, value): self._acct_num = value @acct_num.deleter def del_acct_num(self): del self._acct_num如果想让属性只读,只需要去掉setter方法。
在运行时创建描述符
我们可以在运行时添加property属性:
class Person(object): def addProperty(self, attribute): # create local setter and getter with a particular attribute name getter = lambda self: self._getProperty(attribute) setter = lambda self, value: self._setProperty(attribute, value) # construct property attribute and add it to the class setattr(self.__class__, attribute, property(fget=getter, \ fset=setter, \ doc="Auto-generated method")) def _setProperty(self, attribute, value): print("Setting: {} = {}".format(attribute, value)) setattr(self, _ + attribute, value.title()) def _getProperty(self, attribute): print("Getting: {}".format(attribute)) return getattr(self, _ + attribute) >>> user = Person() >>> user.addProperty(name) >>> user.addProperty(phone) >>> user.name = john smith Setting: name = john smith >>> user.phone = 12345 Setting: phone = 12345 >>> user.name Getting: name John Smith >>> user.__dict__ {_phone: 12345, _name: John Smith}静态方法和类方法
我们可以使用描述符来模拟Python中的@staticmethod和@classmethod的实现。我们首先来浏览一下下面这张表:
Transformation Called from an Object Called from a Class function f(obj, *args) f(*args) staticmethod f(*args) f(*args) classmethod f(type(obj), *args) f(klass, *args)静态方法
对于静态方法f。c.f和C.f是等价的,都是直接查询object.__getattribute__(c, ‘f’)或者object.__getattribute__(C, ’f‘)。静态方法一个明显的特征就是没有self变量。
静态方法有什么用呢?假设有一个处理专门数据的容器类,它提供了一些方法来求平均数,中位数等统计数据方式,这些方法都是要依赖于相应的数据的。但是类中可能还有一些方法,并不依赖这些数据,这个时候我们可以将这些方法声明为静态方法,同时这也可以提高代码的可读性。
使用非数据描述符来模拟一下静态方法的实现:
class StaticMethod(object): def __init__(self, f): self.f = f def __get__(self, obj, objtype=None): return self.f我们来应用一下:
class MyClass(object): @StaticMethod def get_x(x): return x print(MyClass.get_x(100)) # output: 100类方法
Python的@classmethod和@staticmethod的用法有些类似,但是还是有些不同,当某些方法只需要得到类的引用而不关心类中的相应的数据的时候就需要使用classmethod了。
使用非数据描述符来模拟一下类方法的实现:
class ClassMethod(object): def __init__(self, f): self.f = f def __get__(self, obj, klass=None): if klass is None: klass = type(obj) def newfunc(*args): return self.f(klass, *args) return newfunc其他的魔术方法
***接触Python魔术方法的时候,我也被__get__, __getattribute__, __getattr__, __getitem__之间的区别困扰到了,它们都是和属性访问相关的魔术方法,其中重写__getattr__,__getitem__来构造一个自己的集合类非常的常用,下面我们就通过一些例子来看一下它们的应用。
__getattr__
Python默认访问类/实例的某个属性都是通过__getattribute__来调用的,__getattribute__会被无条件调用,没有找到的话就会调用__getattr__。如果我们要定制某个类,通常情况下我们不应该重写__getattribute__,而是应该重写__getattr__,很少看见重写__getattribute__的情况。
从下面的输出可以看出,当一个属性通过__getattribute__无法找到的时候会调用__getattr__。
In [1]: class Test(object): ...: def __getattribute__(self, item): ...: print(call __getattribute__) ...: return super(Test, self).__getattribute__(item) ...: def __getattr__(self, item): ...: return call __getattr__ ...: In [2]: Test().a call __getattribute__ Out[2]: call __getattr__ 应用对于默认的字典,Python只支持以obj[foo]形式来访问,不支持obj.foo的形式,我们可以通过重写__getattr__让字典也支持obj[foo]的访问形式,这是一个非常经典常用的用法:
class Storage(dict): """ A Storage object is like a dictionary except `obj.foo` can be used in addition to `obj[foo]`. """ def __getattr__(self, key): try: return self[key] except KeyError as k: raise AttributeError(k) def __setattr__(self, key, value): self[key] = value def __delattr__(self, key): try: del self[key] except KeyError as k: raise AttributeError(k) def __repr__(self): return <Storage + dict.__repr__(self) + >我们来使用一下我们自定义的加强版字典:
>>> s = Storage(a=1) >>> s[a] 1 >>> s.a 1 >>> s.a = 2 >>> s[a] 2 >>> del s.a >>> s.a ... AttributeError: a__getitem__
getitem用于通过下标[]的形式来获取对象中的元素,下面我们通过重写__getitem__来实现一个自己的list。
class MyList(object): def __init__(self, *args): self.numbers = args def __getitem__(self, item): return self.numbers[item] my_list = MyList(1, 2, 3, 4, 6, 5, 3) print my_list[2]这个实现非常的简陋,不支持slice和step等功能,请读者自行改进,这里我就不重复了。
应用下面是参考requests库中对于__getitem__的一个使用,我们定制了一个忽略属性大小写的字典类。
程序有些复杂,我稍微解释一下:由于这里比较简单,没有使用描述符的需求,所以使用了@property装饰器来代替,lower_keys的功能是将实例字典中的键全部转换成小写并且存储在字典self._lower_keys中。重写了__getitem__方法,以后我们访问某个属性首先会将键转换为小写的方式,然后并不会直接访问实例字典,而是会访问字典self._lower_keys去查找。赋值/删除操作的时候由于实例字典会进行变更,为了保持self._lower_keys和实例字典同步,首先清除self._lower_keys的内容,以后我们重新查找键的时候再调用__getitem__的时候会重新新建一个self._lower_keys。
class CaseInsensitiveDict(dict): @property def lower_keys(self): if not hasattr(self, _lower_keys) or not self._lower_keys: self._lower_keys = dict((k.lower(), k) for k in self.keys()) return self._lower_keys def _clear_lower_keys(self): if hasattr(self, _lower_keys): self._lower_keys.clear() def __contains__(self, key): return key.lower() in self.lower_keys def __getitem__(self, key): if key in self: return dict.__getitem__(self, self.lower_keys[key.lower()]) def __setitem__(self, key, value): dict.__setitem__(self, key, value) self._clear_lower_keys() def __delitem__(self, key): dict.__delitem__(self, key) self._lower_keys.clear() def get(self, key, default=None): if key in self: return self[key] else: return default我们来调用一下这个类:
>>> d = CaseInsensitiveDict() >>> d[ziwenxie] = ziwenxie >>> d[ZiWenXie] = ZiWenXie >>> print(d) {ZiWenXie: ziwenxie, ziwenxie: ziwenxie} >>> print(d[ziwenxie]) ziwenxie # d[ZiWenXie] => d[ziwenxie] >>> print(d[ZiWenXie]) ziwenxie猜你喜欢
- 用粘土手工打造独特电脑装饰(发挥创意,打造个性化电脑装饰品)
 - 在windows 7中不用打开文档也可以看到文档内容的方法介绍
 - windows 7双显示器最大化窗口显示不全怎么办?
 - windows 7/windows 8系统如何查看WIFI密码 其他系统查看方法类
 - 如何使用苹果设备分享WiFi密码给其他手机(简单实用的方法和步骤,让你轻松共享网络连接)
 - windows 7下屏蔽(禁用)鼠标滚轮就是中间那个可以自由滚动的轮
 - 10个重要的Linux ps命令实战
 - ubuntu打开的正确方式(笑)在这个快节奏的时代。能够使用最快速度最低代价完成任务具有非常重要的意义,对于现代开发者来说,Linux是生活中必不可少且不可替代的工具,假如我们在开发少或学习中少一些遇到环境的坑,会节省很多时间,潜伏在各种Linux群中数年,发现大家都有手贱(操作失误,笑),因此在本文中给出终极解决方案: 直接使用Ubuntu Live CD 来完成系统的启动。那么最快的Linux安装办法又是什么呢?其实经过严谨的思考之后我个人觉得是不安装。我们可以准备从grub启动ubuntu操作系统把系统放到内存中,再预制出各种情况的环境包,也就是环境模块化,因为系统是在整个内存中所以速度非常非常快,需要模块化的内容并不多,比如说ssh vim tmux 为一组工具,开机后一个命令就可以激活工具包,不到几秒钟的时间即可完成,其他的如Chrome浏览器一类的都可以快速完成在内存上的安装。通过这种方式让环境更加灵活更加快速准备好开发环境。这是一个思路上的变化,从我需要什么就安装什么一起都准备好给你用->变成了只启动基础的系统然后我需要什么我就后加载什么。最后的准备在于容器,Docker 以及RKT与RunC的准备,在团队内可以秒级推送运行的环境。本文依然有它的缺点: 主要体现在两个方面假如默认驱动不好使会造成很多麻烦。对于内存小的开发机器还是做快照比较好。最少4G内存,我们推荐内存在8G以上最好是16G或者32G内存的机器上采用此方案。因为本方案整个操作系统都是加载到内存中的,硬盘速度即为内存的速度,我这里实测可以达到4.5Gb/s,所以你准备好屌丝逆袭(买不起SSD)秒杀高富帅了吗?ubuntu对于它的简单介绍可以参考这里。当然对于我本人而言选择Ubuntu主要原因仅仅是对他比较熟悉。并不限制使用什么发行版本。假如您要是选本方案搭建环境,并且选择其他发行版本请选择LiveCD尽量小一点的,假如能自己裁剪LiveCD就更好了,另外尽量选择64位架构。当然自己越熟悉越好。虽然Ubuntu有一点不自由,但是我个人觉得包的质量管理还是非常好的,的确适合开发者使用。Docker官方:https://www.docker.com/它对于我来说,几乎就像是水,假如没有他就没有今天的丰富多彩的运行方式,它可以让程序连同环境一起打包运行,使得部署备份等等非常方便,对于本次环境构建而言我们可以在个人的registry上备份自己的环境,只需要一小段时间就可以克隆回来。再加上现在各种永久免费的国内加速服务这种环境部署就是程序员天堂。更多细节请关注这里。在本文中我们使用单文件版本的Docker文件,方便可控。无任何累赘。本人不喜脚本安装。RunC && RKT这两种产品是libcontainer的代表。相对于Docker来说,它是一个容器只需要一个进程,对于Docker来说,他需要一个守护进程。在本文中(本博主)不评价任何产品的好坏,只说什么时候适合用什么。当然在我们这里利用了它启动方便,挂载方便的特性让我们的工作更方便更方便更方便!由于现在RunC还没有发布1.0版本文档几乎没有,因此我这里会给出我的详细研究结果。至于RKT还是大家自己摸索,文档已经很全了。CoreOS作为专门运行容器的操作系统,稳定性可不属性都是前所未有,当然我们的容器可以跑在这个上面本文中会提及两个系统的兼容性方案。Ubuntu适合程序开发的情况。启动速度可能在1分钟左右(有点慢)。但是有图形界面,有定制基础环境的方案。CoreOS适合跑程序,启动速度在10秒以下适合释放更多内存来跑临时业务,做测试的时候实用此方案。从Grub开始什么是Grub它可以装在硬盘上帮助你启动操作系统,几乎所有操作系统都可以,当然也包含本次的方案内容。安装方式有很多种。而且必须得根据自己的情况来安装。因为每个人电脑上的操作系统情况都不一样,有的已经有系统了。有的不想给自己装引导。可能先放到u盘上引导都有可能。因此这里不做安装方法上的更多描述可以自己根据情况到搜索引擎上搜索。Grub 的安装我之前是安装了windows,所以我直接安装了EasyBCD然后安装NeoGRUB来实现的。具体操作步骤如下:下载安装EasyBCD如图所示安装重启电脑之后选择Grub启动项可以进入一个类似输入命令的界面即可确定完成安装。Ubuntu 启动配置启动准备假如想要准备启动Ubuntu那么有三个关键要素,ubuntu-14.04.2-desktop-amd64.isoimage中casper/vmlinuz.efiimage中casper/initrd.lz这三个文件作为启动的必要文件请放到分区中容易找到的地方。推荐做法:给自己的硬盘上开个EXT4 分区,然后把三个文件丢到里面去。启动参数调试工作中我们讲求流程,当然我们也需要一定的路程,我们不可能一次调整就成功把ISO启动起来。因此我们先加一个调试流程。root (hd0,1)kernel (hd0,1)/casper/vmlinuz.efi boot=casper iso-scan/filename=/ubuntu-14.04.2-desktop-amd64.iso ro quiet splash locate=en_US.UTF-8initrd (hd0,1)/casper/initrd.lz你需要看清楚把文件放到了第几块硬盘上,第几块分区上。组成参数为(hd0,1)(我的是0号硬盘第二个分区)。假如你实在分不清可以参考这里。root (hd0,1) 设置GRUB的root设备kernel 命令套格式,变化参数 vmlinuz.efi 和 ubuntu-14.04.2-desktop-amd64.iso 要与你放到硬盘上的位置对应,可以使用Tab自动补全非常实用。initrd 依然是套格式找到对应硬盘傻姑娘的initrd.lz的位置。假如出现错误了按↑会出现刚才的命令,多加修改直到成功启动。启动参数放到硬盘中假如是NeoGrub配置文件路径C:/NST/menu.lst,其他的请参考说明文档,一般都会有的。配置文件写法:timeout 3default 0title Ubunturoot (hd0,1)kernel (hd0,1)/casper/vmlinuz.efi boot=casper iso-scan/filename=/ubuntu-14.04.2-desktop-amd64.iso ro quiet splash locate=en_US.UTF-8initrd (hd0,1)/casper/initrd.lz3秒等待自动选择第一项,名称ubuntu 别的剩下的就是刚才调试内容的命令了。完成刚需配置刚醒需求配置是指在容器之外的刚醒需求配置,比如说编辑器,SSH等等工具箱。配置方法: APT。修改安装源:sed -i ‘s/archive.ubuntu.com/mirrors.aliyun.com/g’ /etc/apt/sources.list && apt-get update安装你需要的软件包:apt-get install -y vim screen tmux ssh-server等等调试你的环境看看是否好使,然后复制文件夹/var/cache/apt/archives中所有deb文件即为您刚才所有的安装包。复制到一个可以持久化的文件夹中等待使用。回调环境: 每当您重新启动之后这些安装包即消失,想恢复,进入到上一步中复制到的文件夹中执行如下命令dpkg -i *即可安装预先准备好的软件灵活使用,此步骤为发挥组合拳威力的一步。当您反复定制凝练您的安装包之后,会根据个人情况选择不同的目录来初始化您的环境,其结果是,又小又轻,非常可靠。再也不怕手贱导致环境崩坏,再加上外部的git服务保证工作内容的安全性基本上可以肯定的说:我们的环境与硬件与工作耦合度降低不少.对于APT无法安装的内容,比如说JDK 与定制版本的Eclipse您可以通过下载完之后持久化保存到硬盘中,写一个脚本解压后放到硬盘中至于环境直接写入到profile中即可。使用此方法之后假如您使用IDE真的会有一个非常非常大的速度提升,我打开Eclipse C/C++ 5秒钟即可。容器安装配置一般情况下我们安装Docker都是通过网上下载脚本安装,但是这种情况下我们推荐使用直接下载单文件版本的来完成安装安装脚本如下:curl -sSL -O https://get.daocloud.io/docker/builds/Linux/x86_64/docker-latestchmod +x docker-latestmv docker-latest /usr/bin/dockerdocker -d &在第三行中复制docker程序到任何Path下的文件夹内备用。在配置上,还是要吧/var/lib/docker放到可持久化的软连接下面。比如说按照下面的操作:rm -rf /var/lib/dockerln -s /isodevice/docker /var/lib/docekr按照这种方法操作可以让docker image 持久化保存在硬盘中。重启也能找回来。这里注意,假如您之前就实践了CoreOS 171.3 版本上的Docker它使用的是1.6版本他的存储文件格式与1.7的不同因此不通用。我就踩到这个坑了。接下来的操作就与Docker一样了。注意:这个安装包: cgroup-lite_1.9_all.deb 一定要装,默认的光盘中就差这一个依赖,在下面的安装包中有要不就apt安装保存下来都可以。材料准备我在百度云盘中准备了三个文件,第一个是docker 1.7.1 第二个准别了vim screen ssh-server tmux CGroup的安装包。最后一个把我常用的Docker images 打包放出来了。其实在准备这个方案花最多时间就是在于精简依赖,最后发现只需要一个CGroup安装包。总结在本段文档中,我们给出了一个全新的使用Linux操作系统的思路不安装。并且做了一定的前期准备工作。启动了Linux操作系统做到本步骤假如您是新手就可以开始Linux的学习旅程了,假如您是需要此方案工作已经可以展开一般性质的工作了。最后我真的拆掉我的三爽垃圾ssd(越用越慢)直接上内存。方案合理要素:root权限过大。有时导致无法挽回的误操作。现在的电脑内存不是很贵。大部分Linux电脑内存都在闲置。Linux操作系统假如每次执行的任务假如比较单一(一台物理机只给一个人用)真的不会很大。给重新启动计算机赋予了新的意义。真正超过SSD的速度,成本更低。(实测启动Eclipse C/C++ 仅需要5s)
 - 电脑共享使用教程