前言
Python 中的特殊方法主要是为了被解释器调用的,因此应该尽量使用 len(my_object) 而不是 my_object.__len__() 这种写法。在执行 len(my_object) 时,Python 解释器会自行调用 my_object 中实现的 __len__ 方法。
除非有大量的元编程存在,直接调用特殊方法的频率应远小于实现它们的次数。
模拟数值类型
可以通过在自定义对象中实现 __add__ 和 __mul__ 等特殊方法 ,令其支持 +、* 等运算符。
如下面的模拟向量的 Vector 类:
# vector.py from math import hypot class Vector: def __init__(self, x=0, y=0): self.x = x self.y = y def __repr__(self): return f'Vector({self.x}, {self.y})' def __abs__(self): return hypot(self.x, self.y) def __bool__(self): return bool(self.x or self.y) def __add__(self, other): return Vector(self.x + other.x, self.y + other.y) def __mul__(self, scalar): return Vector(self.x * scalar, self.y * html" target="_blank">scalar)
运行效果如下:
>>> from vector import Vector
>>> v1 = Vector(2, 4)
>>> v2 = Vector(2, 1)
>>> v1 + v2
Vector(4, 5)
>>> v = Vector(3, 4)
>>> abs(v)
5.0
>>> v * 3
Vector(9, 12)
对象的字符串表示
Python 有一个 repr 内置函数,能把一个对象用字符串的形式表示出来。实际上这种字符串表达是通过对象内部的 __repr__ 特殊方法定义的。默认情况下,在控制台里查看某个对象时,输出的字符串一般是 <xxx object at 0x7fc99d6ab2e0> 这种形式。
__repr__ 返回的字符串应该准确、无歧义,并尽可能表示出该对象是如何创建的。比如前面的 Vector 对象,其 __repr__ 中定义的字符串形式类似于 Vector(3, 4),和对象初始化的语法非常近似。
__repr__ 和 __str__ 的区别在于,__str__ 是在向对象应用 str() 函数(或者用 print 函数打印某个对象)时被调用。其返回的字符串对终端用户更友好。
如果只想实现其中一个特殊方法,__repr__ 应该是更优的选择。在对象没有实现 __str__ 方法的情况下,Python 解释器会用 __repr__ 代替。
# myclass.py class MyClass: def __repr__(self): return 'MyClass' def __str__(self): return 'This is an instance of MyClass'
>>> from myclass import MyClass >>> my = MyClass() >>> my MyClass >>> print(my) This is an instance of MyClass
自定义布尔值
Python 里有 bool 类型,但实际上任何对象都可以用在需要 bool 类型的上下文(比如 if 或 while 语句)中。为了判断某个值 x 的真假,Python 会调用 bool(x) 返回 True 或 False。
默认情况下,自定义类的实例总是为真。除非这个类对于 __bool__ 或 __len__ 方法有自己的实现。
bool(x) 实际上调用了对象 x 中的 __bool__ 方法。如不存在 __bool__ 方法,则 bool(x) 会尝试调用 x.__len__(),返回 0 则为 False,否则为 True。
# boolclass.py class BoolClass: def __init__(self): self.list = [] def add(self, item): self.list.append(item) def __len__(self): return len(self.list)
>>> from boolclass import BoolClass >>> b = BoolClass() >>> len(b) 0 >>> bool(b) False >>> b.add(1) >>> len(b) 1 >>> bool(b) True
# boolclass.py class BoolClass: def __init__(self): self.list = [] def add(self, item): self.list.append(item) def __len__(self): return len(self.list) def __bool__(self): return bool(sum(self.list))
>>> from boolclass import BoolClass >>> b = BoolClass() >>> b.add(1) >>> len(b) 1 >>> bool(b) True >>> b.add(-1) >>> len(b) 2 >>> bool(b) False
参考资料
Fluent Python
总结
到此这篇关于Python中特殊方法以及应用详解的文章就介绍到这了,更多相关Python特殊方法及应用内容请搜索小牛知识库以前的文章或继续浏览下面的相关文章希望大家以后多多支持小牛知识库!
本文向大家介绍Python特殊属性property原理及使用方法解析,包括了Python特殊属性property原理及使用方法解析的使用技巧和注意事项,需要的朋友参考一下 1 什么是特性property property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值 注意:此时的特性arear和perimeter不能被赋值 2 为什么要用property 将一个类的函数定义成特性以后,
本文向大家介绍Golang import 导入包语法及一些特殊用法详解,包括了Golang import 导入包语法及一些特殊用法详解的使用技巧和注意事项,需要的朋友参考一下 package 的导入语法 写 Go 代码的时经常用到 import 这个命令用来导入包,参考如下: 然后在代码里面可以通过如下的方式调用: fmt 是 Go 的标准库,它其实是去 GOROOT 下去加载该模块,当然 Go
本文向大家介绍列举面向对象中的特殊成员以及应用场景相关面试题,主要包含被问及列举面向对象中的特殊成员以及应用场景时的应答技巧和注意事项,需要的朋友参考一下
目录表 特殊的方法 单语句块 列表综合 使用列表综合 在函数中接收元组和列表 lambda形式 使用lambda形式 exec和eval语句 assert语句 repr函数 概括 到目前为止,我们已经学习了绝大多数常用的Python知识。在这一章中,我们将要学习另外一些方面的Python知识,从而使我们对Python的了解更加 完整 。 特殊的方法 在类中有一些特殊的方法具有特殊的意义,比如__i
本文向大家介绍Python字典简介以及用法详解,包括了Python字典简介以及用法详解的使用技巧和注意事项,需要的朋友参考一下 老规矩以下方法环境2.7.x,请3.x以上版本的朋友记得格式print(输出内容放入括号内) 字典的基本组成以及用法 首先来说说字典是由key键与value值一一对应来组成字典的基本结构 key键不能由list列表,dict字典等多元素命名, key是唯一属性又可以称一对
问题内容: 我希望有人能够回答这个对Python有深刻理解的问题 考虑以下代码: 注意)如何不会产生的预期结果?我想知道为什么会这样,是否有办法实现这一目标… 相比之下,下面的示例仍然有效(也许是因为我们不打算重写特殊方法吗?): 问题答案: 不会调用特殊方法,那些__在实例上带有名称包围的特殊方法,而仅在类上,显然可以提高性能。因此,无法直接在实例上进行覆盖并使其起作用。相反,你需要执行以下操作