流畅的python第十九章元编程学习记录
在 Python 中,数据的属性和处理数据的方法统称属性(attribute)。其实,方法只是可调
用的属性。除了这二者之外,我们还可以创建特性(property),在不改变类接口的前提
下,使用存取方法(即读值方法和设值方法)修改数据属性
除了特性,Python 还提供了丰富的 API,用于控制属性的访问权限,以及实现动态属性。
使用点号访问属性时(如 obj.attr),Python 解释器会调用特殊的方法(如
__getattr__ 和 __setattr__)计算属性
使用 __new__ 方法以灵活的方式创建对象
我们通常把 __init__ 称为构造方法,这是从其他语言借鉴过来的术语。其实,用于构建
实例的是特殊方法 __new__:这是个类方法(使用特殊方式处理,因此不必使用
@classmethod 装饰器),必须返回一个实例。返回的实例会作为第一个参数(即
self)传给 __init__ 方法。因为调用 __init__ 方法时要传入实例,而且禁止返回任何
值,所以 __init__ 方法其实是“初始化方法”。真正的构造方法是 __new__。我们几乎不
需要自己编写 __new__ 方法,因为从 object 类继承的实现已经足够了。
创建可读写特性
class LineItem:
def __init__(self, description, weight, price):
self.description = description
self.weight = weight
self.price = price
def subtotal(self):
return self.weight * self.price
@property
def weight(self):
return self.__weight
@weight.setter
def weight(self, value):
if value > 0:
self.__weight = value
else:
raise ValueError('value must be > 0')
去除重复的方法是抽象。抽象特性的定义有两种
方式:使用特性工厂函数,或者使用描述符类。后者更灵活
虽然内置的 property 经常用作装饰器,但它其实是一个类。在 Python 中,函数和类通
常可以互换,因为二者都是可调用的对象,而且没有实例化对象的 new 运算符,所以调
用构造方法与调用工厂函数没有区别。此外,只要能返回新的可调用对象,代替被装饰的
函数,二者都可以用作装饰器。
不适用property装饰器的例子,经典的调用
class LineItem:
def __init__(self, description, weight, price):
self.description = description
self.weight = weight
self.price = price
def subtotal(self):
return self.weight * self.price
def get_weight(self):
return self.__weight
def set_weight(self, value):
if value > 0:
self.__weight = value
else:
raise ValueError('value must be > 0')
weight = property(get_weight, set_weight)
某些情况下,这种经典形式比装饰器句法好;稍后讨论的特性工厂函数就是一例。但是,
在方法众多的类定义体中使用装饰器的话,一眼就能看出哪些是读值方法,哪些是设值方
法,而不用按照惯例,在方法名的前面加上 get 和 set。
本节的主要观点是,obj.attr 这样的表达式不会从 obj 开始寻找 attr,而是从
obj.__class__ 开始,而且,仅当类中没有名为 attr 的特性时,Python 才会在 obj 实
例中寻找。这条规则不仅适用于特性,还适用于一整类描述符——覆盖型描述符
先寻找类属性,在寻找实例属性
如果使用经典调用句法,为 property 对象设置文档字符串的方法是传入 doc 参数:
weight = property(get_weight, set_weight, doc='weight in kilograms')
使用装饰器创建 property 对象时,读值方法(有 @property 装饰器的方法)的文档字
符串作为一个整体,变成特性的文档
创建特性工厂函数
def quantity(storage_name):
def qty_getter(instance):
return instance.__dict__[storage_name]
def qty_setter(instance, value):
if value > 0:
instance.__dict__[storage_name] = value
else:
raise ValueError('value must be > 0')
return property(qty_getter, qty_setter)
class LineItem:
weight = quantity('weight')
price = quantity('price')
def __init__(self, description, weight, price):
self.description = description
self.weight = weight
self.price = price
def subtotal(self):
return self.weight * self.price
在真实的系统中,分散在多个类中的多个字段可能要做同样的验证,此时最好把
quantity 工厂函数放在实用工具模块中,以便重复使用。最终可能要重构那个简单的工
厂函数,改成更易扩展的描述符类,然后使用专门的子类执行不同的验证。
处理属性删除操作
class BlackKnight:
def __init__(self):
self.members = ['an arm', 'another arm', 'a leg', 'another leg']
self.phrases = ["It's but a scratch.",
"It's just a flesh wound.",
"I'm invincible!",
"All right, we'll call it a draw"]
@property
def member(self):
print('next member is:')
return self.members[0]
@member.deleter
def member(self):
text = 'BLACK KNIGHT (loses {})\n -- {}'
print(text.format(self.members.pop(0), self.phrases.pop(0)))
影响属性处理方式的特殊属性
后面几节中的很多函数和特殊方法,其行为受下述 3 个特殊属性的影响。
__class__
对象所属类的引用(即 obj.__class__ 与 type(obj) 的作用相同)。Python 的某些
特殊方法,例如 __getattr__,只在对象的类中寻找,而不在实例中寻找。
__dict__
一个映射,存储对象或类的可写属性。有 __dict__ 属性的对象,任何时候都能随意
设置新属性。如果类有 __slots__ 属性,它的实例可能没有 __dict__ 属性。参见下面
对 __slots__ 属性的说明。
__slots__
类可以定义这个这属性,限制实例能有哪些属性。__slots__ 属性的值是一个字符
串组成的元组,指明允许有的属性。 如果 __slots__ 中没有 '__dict__',那么该类
的实例没有 __dict__ 属性,实例只允许有指定名称的属性。
dir([object])
列出对象的大多数属性。官方文档
(https://docs.python.org/3/library/functions.html#dir)说,dir 函数的目的是交互式使用,
因此没有提供完整的属性列表,只列出一组“重要的”属性名。dir 函数能审查有或没有
__dict__ 属性的对象。dir 函数不会列出 __dict__ 属性本身,但会列出其中的
键。dir 函数也不会列出类的几个特殊属性,例如 __mro__、__bases__ 和 __name__。
如果没有指定可选的 object 参数,dir 函数会列出当前作用域中的名称。
getattr(object, name[, default])
从 object 对象中获取 name 字符串对应的属性。获取的属性可能来自对象所属的类
或超类。如果没有指定的属性,getattr 函数抛出 AttributeError 异常,或者返回
default 参数的值(如果设定了这个参数的话)。
hasattr(object, name)
如果 object 对象中存在指定的属性,或者能以某种方式(例如继承)通过 object
对象获取指定的属性,返回 True。文档
(https://docs.python.org/3/library/functions.html#hasattr)说道:“这个函数的实现方法是调
用 getattr(object, name) 函数,看看是否抛出 AttributeError 异常。”
setattr(object, name, value)
把 object 对象指定属性的值设为 value,前提是 object 对象能接受那个值。这个
函数可能会创建一个新属性,或者覆盖现有的属性。
vars([object])
返回 object 对象的 __dict__ 属性;如果实例所属的类定义了 __slots__ 属性,
实例没有 __dict__ 属性,那么 vars 函数不能处理那个实例(相反,dir 函数能处理这
样的实例)。如果没有指定参数,那么 vars() 函数的作用与 locals() 函数一样:返回
表示本地作用域的字典。
使用点号或内置的 getattr、hasattr 和 setattr 函数存取属性都会触发下述列表中相
应的特殊方法。但是,直接通过实例的 __dict__ 属性读写属性不会触发这些特殊方法
——如果需要,通常会使用这种方式跳过特殊方法。
不管是使用点号存取属性,还是使用 19.6.2 节列出的某个内置函数,都会触发下述特殊方
法中的一个。例如,obj.attr 和 getattr(obj, 'attr', 42) 都会触发
Class.__getattribute__(obj, 'attr') 方法。
__delattr__(self, name)
只要使用 del 语句删除属性,就会调用这个方法。例如,del obj.attr 语句触发
Class.__delattr__(obj, 'attr') 方法。
__dir__(self)
把对象传给 dir 函数时调用,列出属性。例如,dir(obj) 触发
Class.__dir__(obj) 方法。
__getattr__(self, name)
仅当获取指定的属性失败,搜索过 obj、Class 和超类之后调用。表达式
obj.no_such_attr、getattr(obj, 'no_such_attr') 和 hasattr(obj,
'no_such_attr') 可能会触发 Class.__getattr__(obj, 'no_such_attr') 方法,但
是,仅当在 obj、Class 和超类中找不到指定的属性时才会触发。
__getattribute__(self, name)
尝试获取指定的属性时总会调用这个方法,不过,寻找的属性是特殊属性或特殊方法
时除外。点号与 getattr 和 hasattr 内置函数会触发这个方法。调用
__getattribute__ 方法且抛出 AttributeError 异常时,才会调用 __getattr__ 方
法。为了在获取 obj 实例的属性时不导致无限递归,__getattribute__ 方法的实现要
使用 super().__getattribute__(obj, name)。
__setattr__(self, name, value)
尝试设置指定的属性时总会调用这个方法。点号和 setattr 内置函数会触发这个方
法。例如,obj.attr = 42 和 setattr(obj, 'attr', 42) 都会触发
Class.__setattr__(obj, ‘attr’, 42) 方法。
流畅的python第十九章元编程学习记录的更多相关文章
- 流畅的python第十六章协程学习记录
从句法上看,协程与生成器类似,都是定义体中包含 yield 关键字的函数.可是,在协程中,yield 通常出现在表达式的右边(例如,datum = yield),可以产出值,也可以不产出——如果 yi ...
- 流畅的python第十四章可迭代的对象,迭代器和生成器学习记录
在python中,所有集合都可以迭代,在python语言内部,迭代器用于支持 for循环 构建和扩展集合类型 逐行遍历文本文件 列表推导,字典推导和集合推导 元组拆包 调用函数时,使用*拆包实参 本章 ...
- 流畅的python第十五章上下文管理器和else块学习记录
with 语句和上下文管理器for.while 和 try 语句的 else 子句 with 语句会设置一个临时的上下文,交给上下文管理器对象控制,并且负责清理上下文.这么做能避免错误并减少样板代码, ...
- 流畅的python第十二章继承的优缺点学习记录
子类化内置类型的缺点 多重集成和方法解析顺序 tkinter
- 高性能MySQL之【第十五章 备份与恢复】学习记录
我们不打算包括的话题: 安全(访问备份,恢复数据的权限,文件是否需要加密) 备份存储在哪里,包括他们应该离源数据多远,以及如何将数据从源头移动到目的地 保留策略.审计 ...
- Python之路【第十九章】:Django进阶
Django路由规则 1.基于正则的URL 在templates目录下创建index.html.detail.html文件 <!DOCTYPE html> <html lang=&q ...
- python 教程 第十九章、 图形界面编程
第十九章. 图形界面编程 import Tkinter top = Tkinter.Tk() hello = Tkinter.Label(top, text='Hello World!') hello ...
- 第十九章 Django的ORM映射机制
第十九章 Django的ORM映射机制 第一课 Django获取多个数据以及文件上传 1.获取多选的结果(checkbox,select/option)时: req.POST.getlist('fav ...
- 第十九章——使用资源调控器管理资源(1)——使用SQLServer Management Studio 配置资源调控器
原文:第十九章--使用资源调控器管理资源(1)--使用SQLServer Management Studio 配置资源调控器 本系列包含: 1. 使用SQLServer Management Stud ...
随机推荐
- mysql-表完整性约束
阅读目录 一 介绍 二 not null与default 三 unique 四 primary key 五 auto_increment 六 foreign key 七 总结 一 介绍 回到顶 ...
- 通过百度地图API获取经纬度以及两点间距离
package com.baidumap; import java.io.BufferedReader; import java.io.IOException; import java.io.Inpu ...
- nginx应用场景,特性,目录结构,常用模块,内置变量,URL和URI,http状态码,配置文件详解
1.nginx介绍 1丶俄罗斯人开发的,开源www服务软件 2丶软件一共780K 3丶nginx本身是一款静态(html,js,css,jpg等)www软件 4丶静态小文件高并发,同时占用的资源很少, ...
- 《深入理解Android2》读书笔记(一)
2017-5-12 从今天开始估计有一段空闲时间,开始阅读<深入理解Android2>,并写读书笔记. 第一章搭建环境直接略过. 第二章是Binder,暂时略过 7大类服务包括:1.And ...
- 洛谷P2587 [ZJOI2008] 泡泡堂
题目传送门 分析:一道策略游戏题,要求最大期望得分和最小期望得分.首先分析最大,很显然是可以用一种类似于田忌赛马的思维来做,将两队的实力按照从大到小(其实从小到大也可以)排序,然后就按照顺序比较,可能 ...
- Python3 字典及三级菜单练习
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # Author;Tsukasa list_1 = { '广州':{ '越秀区':{ '五羊石像','镇海 ...
- 【读书笔记】周志华《机器学习》第三版课后习题讨<第一章-绪论>
虽然是绪论..但是...真的有点难!不管怎么说,一点点前进吧... 声明一下答案不一定正确,仅供参考,为本人的作答,希望大神们能多多指教~ 1.1 表1.1中若只包含编号为1和4的两个样例,试给出相应 ...
- FFT实现高精度乘法
你应该知道$FFT$是用来处理多项式乘法的吧. 那么高精度乘法和多项式乘法有什么关系呢? 观察这样一个$20$位高精度整数$11111111111111111111$ 我们可以把它处理成这样的形式:$ ...
- Python开发基础-Day3-列表、元组和字典
列表 列表定义:[]内以逗号分隔,按照索引,存放各种数据类型,每个位置代表一个元素 特性: 1.可存放多个值 2.可修改指定索引位置对应的值,可变 3.按照从左到右的顺序定义列表元素,下标从0开始顺序 ...
- UML类图—机房收费系统
UML类图:显示了系统的静态结构,而系统的静态结构构成了系统的概念基础.类图用于对系统中的各种概念进行建模,并描绘他们之间的关系.在类图中,一共包含了一下集中模型元素,分别是:类.接口.依赖关系.关联 ...