python风格对象
对象表示形式
python提供了两种获取对象字符串表示形式的标准方式
repr() //便于开发者理解的方式返回对象的字符串表示形式(一般来说满足obj==eval(repr(obj)))
str() //便于用户理解的方式返回对象的字符串表示形式
要使对象能这两种内置函数的参数,需要实现__repr__和__str__特殊方法,为repr()和str()提供支持。为了给对象提供其他表示形式,还会用到__bytes__和__format__
bytes() //获取对象字节序列表示形式
format() //特殊格式显示对象字符串表示
构建一个向量类:
from array import array
import math class Vector:
typecode = 'd' def __init__(self, x, y):
self.x = float(x)
self.y = float(y) def __iter__(self): #将类实例变为可迭代对象
return (i for i in (self.x, self.y)) def __repr__(self): #构成供开发者使用的字符串
class_name = type(self).__name__
return '{}({!r}, {!r})'.format(class_name, *self) def __str__(self): #构成供用户使用的字符串
return str(tuple(self)) def __bytes__(self): #将对象实例转为字节序列
return (bytes([ord(self.typecode)]) + bytes(array(self.typecode, self))) def __eq__(self, other): #实现 ==
return tuple(self) == tuple(other) def __abs__(self): #计算模长
return math.hypot(self.x, self.y) def __bool__(self): #零向量
return bool(abs(self))
使用:
if __name__ == '__main__':
V = Vector(3, 4)
print(V.x, V.y)
x, y = V #是可迭代对象,故可以元组拆包
print((x, y))
repr_V = repr(V) #字符串表示
print(repr_V)
print(eval(repr_V) == V) #执行这个字符串,打印结果
octets = bytes(V)
print(octets)
print(abs(V)) #打印模长
print((bool(V), bool(Vector(0, 0)))) #零向量bool返回False 3.0 4.0
(3.0, 4.0)
Vector(3.0, 4.0)
True
b'd\x00\x00\x00\x00\x00\x00\x08@\x00\x00\x00\x00\x00\x00\x10@'
5.0
(True, False)
classmethod与staticmethod
在上例中,可以使用bytes()将对象实例转化为字节序列:
def __bytes__(self): #将对象实例转为字节序列
return (bytes([ord(self.typecode)]) + bytes(array(self.typecode, self)))
现使用classmethod装饰器来实现字节序列转对象实例的方法:
@classmethod
def frombytes(cls, octets):
typecode = (chr(octets[0]))
memv = memoryview(octets[1:]).cast(typecode) #使用传入的字节序列创建视图,使用typecode转换
return cls(*memv)
使用:
vec = Vector.frombytes(octets) #调用者是vector类
print(vec) (3.0, 4.0)
classmethod是类的方法,而不是实例的方法,它改变了调用方法的方式,因此类方法的第一个参数是类本身而不是实例。(类似c++静态方法)
而staticmethod装饰器也改变方法调用方式,但第一个参数不是特殊的值。其实,静态方法就是普通的函数。
class Demo:
@classmethod
def klassmeth(*args):
return args #返回全位置参数 @staticmethod
def statmeth(*args):
return args #返回全位置参数 if __name__ == '__main__':
print(Demo.klassmeth())
print(Demo.klassmeth('spam'))
print(Demo.statmeth())
print(Demo.statmeth('spam')) #结果
(<class '__main__.Demo'>,)
(<class '__main__.Demo'>, 'spam') #无论如何调用,第一个参数始终是Demo类
()
('spam',) #行为类似于普通函数
格式化显示
内置format()函数和str.format()方法把各个类型的格式化方式委托给相应的.__format__(format_spec)方法。format_spec是格式说明符,它是以下之一:
1.format(my_obj,format_spec)的第二个参数
2.str.format()方法的格式字符串,{}里代换字段中冒号的部分
使用示例:
brl = 1/2.43
print(brl)
form1 = format(brl, '0.4f') #使用前者,格式说明符是0.4f
form2 = '1 BRL = {rate:0.2f} USE'.format(rate=brl) #使用后者,代替冒号部分,格式说明符是0.2f
print(form1)
print(form2) 0.4115226337448559
0.4115
1 BRL = 0.41 USE
格式规范语言为一些内置类型提供了专用的表示代码,比如b表示二进制int类型,x表示十六进制int类型,f表示小数形式的float类型,%表示百分数形式。
print(format(42, 'b'))
print(format(42, 'x'))
print(format(2/3, '0.1%'))
print(format(2/3, '0.3f')) 101010
2a
66.7%
0.667
用户可自行定义__format__方法,如果没有,会调用__str__方法返回值。而未定义__format__方法又传入格式说明符作为参数,将抛出TypeError。
实现可散列的对象
要把类实例变为可散列的对象,必须实现__hash__方法和__eq__方法,而__hash__方法需要保证类对象散列值不变。例如Vector类,则需要让x,y是只可读类型。
class Vector:
typecode = 'd' def __init__(self, x, y):
self.__x = float(x) #使用双下划线把属性标记为私有
self.__y = float(y) @property
def x(self):
return self.__x #使用property装饰器将读值方法标记为特性,即可以使用obj.x获取x @property
def y(self):
return self.__y #同x
之后添加__hash__方法就可以将向量变为可散列的:
def __hash__(self):
return hash(self.x) ^ hash(self.y)
实际上只要能够正确实现__hasn__和__eq__方法并且保证实例散列值不会变化即可。
私有属性和保护属性
python不能用private修饰符创建私有属性,但python有个简单机制避免子类覆盖私有属性。
例如,有人编写了Dog类,用到了mood实例属性却未开放,这时你创建了Dog的子类Beagle,如果你在毫不知情的情况下创建了mood实例属性,那么继承的方法就会覆盖掉Dog中的mood属性。出现了问题却难以发现。
为避免这种情况,如果以__mood命名(前面加双下划线,尾部最多有一个下划线)命名实例属性,那么python会把属性名存入__dict__属性中,而且会在前面加一个下划线和类名。对于上例来说父类会变为_Dog__mood而子类会变为_Beagle__mood。这个语言特性叫做名称改写。
对于向量类的实例:
print(V.__dict__)
{'_Vector__x': 3.0, '_Vector__y': 4.0}
它的目的是避免意外访问,却不能防止故意访问(做坏事),任何人都能直接读取私有属性并且给它赋值。
V._Vector__x = 5.0
print(V) #结果
(5.0, 4.0)
也就是说,它是"私有"却不是真正的私有,不可变也不是真正的不可变。
同时,有人将单划线(_xxx)称为保护属性。
__solts__
默认情况下,python在各个实例中名为__dict__的字典里存储实例属性。但由于字典底层使用散列表实现,速度快也耗费大量内存。若处理上百万个属性不多的实例,可通过__slots__属性可节省大量内存,它将使用元组而不是字典来存储实例属性。
子类不能继承父类的__slots__属性,只会使用自己类中定义的。
方式:创建一个类属性__slots__,将它的值设置为一个字符串构成的可迭代对象,其中各个元素表示各个实例属性。一般使用元组:
class Vector:
__slots__ = ('__x', '__y')
这个属性定义是为了告诉解释器:这个类中所以的实例属性都保存在这。这样,python会在各个实例中使用类似元组的结构存储实例变量。
实例只能拥有__slots__列出的属性,除非把__dict__属性加入其中(但这样做 失去了节省内存的功效)
如果定义了类的__slots__属性,此时想把实例作为弱引用的目标,需要把__weakref__添加到__slots__属性中。
覆盖类属性
类属性可以为实例属性提供默认值。Vector类中使用self.typecode读取类属性的值,实例本身没有类属性,self.typecode获取的是类属性Vector.typecode的值。但是如果为不存在的实例属性赋值,就会新建实例属性。
if __name__ == '__main__':
v1 = Vector(1.1, 2.2)
dumpd = bytes(v1)
print(dumpd)
v1.typecode = 'f' #双精度浮点数表示分量
dumpf = bytes(v1)
print(dumpf)
print(Vector.typecode) #结果
b'd\x9a\x99\x99\x99\x99\x99\xf1?\x9a\x99\x99\x99\x99\x99\x01@'
b'f\xcd\xcc\x8c?\xcd\xcc\x0c@'
d #类属性没有变
也可修改类属性来修改所有实例的typecode默认值
Vector.typecode = 'f'
一般使用方式是创建一个子类,在子类中覆盖掉类属性。
以上来自《流畅的python》
python风格对象的更多相关文章
- 第9章 符合Python风格的对象
#<流畅的Python>读书笔记 # 第9章 符合Python风格的对象 # 本章包含以下话题: # 支持用于生成对象其他表示形式的内置函数(如repr().bytes(),等等) # 使 ...
- Fluent_Python_Part4面向对象,09-pythonic-obj,Python风格的对象
第四部分第9章,Python风格的对象 这一章接第1章,说明常见的特殊方法实现. 本章包括以下话题: 支持用于生成对象其它表示形式的内置函数(如repr().bytes(),等等) 使用一个类方法实现 ...
- Python风格规范
Python风格规范 分号 Tip 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 Tip 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. Py ...
- PYTHON风格规范-Google 开源项目风格指南
Python风格规范 分号 Tip 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 Tip 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. Py ...
- google的Python风格规范
Python风格规范 分号 Tip 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 Tip 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. ...
- Python编码规范和Python风格规范
一.原因 1.长期的工作中,发现大多数程序员的代码可读性差 2.不同的程序员之间的协作很重要,代码可读性必须很好 3.版本升级时,要基于源码升级 4.不友好的代码会影响python的执行效率 二.基于 ...
- Python面对对象相关知识总结
很有一段时间没使用python了,前两天研究微信公众号使用了下python的django服务,感觉好多知识都遗忘了,毕竟之前没有深入的实践,长期不使用就忘得快.本博的主要目的就是对Python中我认为 ...
- Python风格规范-FYI
Python风格规范 分号 Tip 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 Tip 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. Py ...
- Python风格规范分享
今天给大家分享Python 风格规范,以下代码中 Yes 表示推荐,No 表示不推荐. 分号 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 每行不超过80个字符 以下情况除外: 长的 ...
随机推荐
- NOIP 模拟 9 数颜色
题解 一道裸的数据结构题 正解是排序 \(+\) 二分,但是这怎么能有动态开点线段树好写呢? 于是我就打了暴力,骗了五十分. 对于每种颜色,我们在下标上开一颗线段树,对于交换若颜色相同则跳过,否则直接 ...
- NOIP 模拟 $21\; \rm Park$
题解 \(by\;zj\varphi\) 首先,分析一下这个答案:本质上是求在一条路径上,选择了一些点,这些点的贡献是它周围的点权和 - 它上一步的点权 对于一棵树,可以先确定一个根,然后每条路径就可 ...
- java实现随机字母数字验证码
生成随街验证码 VerifyCode 工具类 package com.meeno.common.cerifycode; import javax.imageio.ImageIO; import jav ...
- 如何在WPF中定义窗体模板
参考网址:https://www.cnblogs.com/chenxizhang/archive/2010/01/10/1643676.html可以在app.xaml中定义一个ControlTempl ...
- Object 的wait()方法
The java.lang.Object.wait() causes current thread to wait until another thread invokes the notify() ...
- SpringBoot 优雅配置跨域多种方式及Spring Security跨域访问配置的坑
前言 最近在做项目的时候,基于前后端分离的权限管理系统,后台使用 Spring Security 作为权限控制管理, 然后在前端接口访问时候涉及到跨域,但我怎么配置跨域也没有生效,这里有一个坑,在使用 ...
- opencv入门系列教学(四)处理鼠标事件
一.鼠标事件的简单演示 opencv中的鼠标事件,值得是任何与鼠标相关的任何事物,例如左键按下,左键按下,左键双击等.我们先来看看鼠标事件有哪些,在python中执行下面代码: import cv2 ...
- ES6扩展——函数扩展之默认参数
1.函数的默认参数 //函数的默认参数 function add(a, b = 999){ console.log(a,b); //1 999 } add(1); 2. 函数的形参可以设置默认值,默认 ...
- iptables开启后造成本地套接字阻塞的问题
前段时间,我使用iptables实现了针对IP地址与MAC地址的白名单功能,即将INPUT链的默认规则设为DROP: iptables -P INPUT DROP 这样就能拒绝一切外来报文.随后只需要 ...
- 痞子衡嵌入式:MCUXpresso IDE下将应用程序RW段分散链接的几种方法
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是MCUXpresso IDE下将应用程序RW段分散链接的几种方法. 早期的 MCU 芯片,一般都会嵌入内部 Flash 和 RAM,并且 ...