本文将揭开Python中魔术背后的方法。
微信搜索关注《Python学研大本营》,加入读者群,分享更多精彩
简介
在Python中,魔术方法可以帮助模拟内置函数在Python类中的行为。这些方法的名称前后都有双下划线(__
),因此也被称为双下划线(dunder)方法。
这些魔术方法还可以帮助在Python中实现运算符重载。简单列举一些常见的例子,比如使用乘法运算符*
将两个整数相乘得到乘积。而将它与一个字符串和一个整数k
结合使用时,则会将该字符串重复k
次:
>>> 3 * 4
12
>>> 'code' * 3
'codecodecode'
在这里,我们将通过创建一个简单的二维向量Vector2D
类来探索Python中的魔术方法。
我们将从你可能熟悉的方法开始,并逐渐过渡到更有帮助的魔术方法。
接下来我们开始编写一些魔术方法吧!
1、__init__
考虑以下的Vector2D
类:
class Vector2D:
pass
当创建一个类并实例化一个对象时,可以像这样添加属性:obj_name.attribute_name = value
。
不过,与其在创建的每个实例中手动添加属性,还不如在实例化对象时用一种方法来初始化这些属性。
为此,我们可以定义__init__
方法。接下来让我们为Vector2D
类定义__init__
方法:
class Vector2D:
def __init__(self, x, y):
self.x = x
self.y = y
v = Vector2D(3, 5)
2、__repr__
当你尝试检查或打印实例化的对象时,会发现并没有得到任何有用的信息。
v = Vector2D(3, 5)
print(v)
Output >>> <__main__.Vector2D object at 0x7d2fcfaf0ac0>
这就是为什么应该添加一个表示字符串,即对象的字符串表示形式。为此,可以添加一个__repr__
方法,如下所示:
class Vector2D:
def __init__(self, x, y):
self.x = x
self.y = y
def __repr__(self):
return f"Vector2D(x={self.x}, y={self.y})"
v = Vector2D(3, 5)
print(v)
Output >>> Vector2D(x=3, y=5)
__repr__
应包含创建类的实例所需的所有属性和信息。__repr__
方法通常用于调试。
3、__str__
__str__
也用于添加对象的字符串表示形式。一般来说,__str__
方法用于向类的最终用户提供信息。
接下来为我们的类添加一个__str__
方法:
class Vector2D:
def __init__(self, x, y):
self.x = x
self.y = y
def __str__(self):
return f"Vector2D(x={self.x}, y={self.y})"
v = Vector2D(3, 5)
print(v)
Output >>> Vector2D(x=3, y=5)
如果没有__str__
的实现,它就会回退到__repr__
。因此,对于你创建的每个类,至少应该添加一个__repr__
方法。
4、__eq__
接下来,让我们添加一个方法来检查任意两个Vector2D
类的对象是否相等。如果两个向量对象具有完全相同的x
和y
坐标,则它们是相等的。
现在创建两个x
和y
坐标值均相等的Vector2D
对象,并比较它们是否相等:
v1 = Vector2D(3, 5)
v2 = Vector2D(3, 5)
print(v1 == v2)
结果是False
。因为默认情况下,“比较”会检查的是内存中对象的ID是否相等。
Output >>> False
接下来添加__eq__
方法来检查是否相等:
class Vector2D:
def __init__(self, x, y):
self.x = x
self.y = y
def __repr__(self):
return f"Vector2D(x={self.x}, y={self.y})"
def __eq__(self, other):
return self.x == other.x and self.y == other.y
现在,相等性检查应该可以按预期工作:
v1 = Vector2D(3, 5)
v2 = Vector2D(3, 5)
print(v1 == v2)
Output >>> True
5、__len__
Python的内置len()
函数可以帮助计算内置可迭代对象的长度。对于一个向量来说,长度应该返回向量包含的元素数量。
因此,为Vector2D
类添加一个__len__
方法:
class Vector2D:
def __init__(self, x, y):
self.x = x
self.y = y
def __repr__(self):
return f"Vector2D(x={self.x}, y={self.y})"
def __len__(self):
return 2
v = Vector2D(3, 5)
print(len(v))
所有的Vector2D
类对象的长度都是2
:
Output >>> 2
推荐书单
《Python从入门到精通(第3版)》
《Python从入门到精通(第3版)》从初学者角度出发,通过通俗易懂的语言、丰富多彩的实例,详细介绍了使用Python进行程序开发应该掌握的各方面技术。全书共分27章,包括初识Python、Python语言基础、运算符与表达式、流程控制语句、列表和元组、字典和集合、字符串、Python中使用正则表达式、函数、面向对象程序设计、模块、文件及目录操作、操作数据库、使用进程和线程、网络编程、异常处理及程序调试、Pygame游戏编程、推箱子游戏、网络爬虫开发、火车票分析助手、数据可视化、京东电商销售数据分析与预测、Web编程、Flask框架、e起去旅行网站、Python自动化办公、AI图像识别工具等内容。书中所有知识都结合具体实例进行介绍,涉及的程序代码都给出了详细的注释,读者可轻松领会Python程序开发的精髓,快速提升开发技能。
《Python从入门到精通(第3版)》https://item.jd.com/14055900.html
精彩回顾
《文本抓取利器,Python和Beautiful Soup爬虫助你事半功倍》
《基于Python和Surprise库,新手也能动手搭建推荐系统》
微信搜索关注《Python学研大本营》,加入读者群,分享更多精彩
访问【IT今日热榜】,发现每日技术热点