- Python中的魔术方法是由官方定义好的,它们会在合适的时机被自动调用。 魔法方法的
- 形式:以两个下划线开始并以两个下划线结束。举例:
__init__
点击查看上一篇:Python的魔术方法(魔法方法)(一)
本篇介绍运算相关,算术运算符和类型转换的相关魔法方法。
比较运算符
class Person:
def __init__(self, name, age):
self.name = name
self.age = age
per1 = Person('猪八戒', 18)
per2 = Person('猪八戒', 18)
per3 = Person('孙悟空', 20)
以上述代码为例,我们用Person类创建了三个实例对象:per1
、per2
、per3
。
当我们想看看这些是不是同一个对象,这时候我们会用到==
运算符。
如下面的代码:
print(per1 == per2) # False
print(per1 == per3) # False
print(per2 == per3) # False
很明显,这三个对象分别指向不同的内存地址。
实际上,在对实例进行==
运算符时,python会自动调用__eq__
方法,默认为比较两个对象的内存地址。
而实际上,我们的需求可能是当name
和age
分别对应相等时,就返回True
,否则返回False
,我们可以通过对类的__eq__
方法进行重写来实现我们的需求。
class Person:
def __init__(self, name, age):
self.name = name
self.age = age
def __eq__(self, other):
return self.name == other.name and self.age == other.age
per1 = Person('猪八戒', 18)
per2 = Person('猪八戒', 18)
per3 = Person('孙悟空', 20)
# 此时,对per1和per2的比较返回值就是True了。
print(per1 == per2) # True
print(per1 == per3) # False
同理,当我们对实例间进行不同的比较时,会默认调用以下方法:
比较运算符 | 调用的方法 |
---|---|
per1 == per2 | __eq__ |
per1 != per2 | __ne__ |
per1 > per2 | __gt__ |
per1 < per2 | __lt__ |
per1 >= per2 | __ge__ |
per1 <= per2 | __le__ |
其实,在没有对__ne__
方法进行重写前,__ne__
方法默认结果为对__eq__
结果取反。当然,如果您已经对__ne__
方法进行重写,会直接调用您重写的__ne__
方法。
算术运算符
class Person:
def __init__(self, name, height):
self.name = name
self.age = height
per1 = Person('猪八戒', 175)
print(per1 + 2)
对于上述第8行代码,在程序运行时将会报错,因为此时程序会直接调用__add__
方法。
假设我们的实际需求是增加实例的身高,我们可以通过对__add__
方法进行重写来实现我们的需求。
class Person:
def __init__(self, name, height):
self.name = name
self.height = height
def __add__(self, other):
return self.height + other
per1 = Person('猪八戒', 175)
print(per1 + 2) # 177
通过对__add__
方法的重写,实现对实例直接使用+
运算符,直接对实例的age
属性加。
同样的,还有-
,*
,/
,%
,&&
。以下是各运算符对应的方法:
算术运算法 | 对应的方法 |
---|---|
+ | __add__ |
- | __sub__ |
* | __mul__ |
/ | __truediv__ |
% | __mod__ |
** | __pow__ |
类型转换
class Person:
def __init__(self, name, height):
self.name = name
self.age = height
per1 = Person('猪八戒', 175)
print(str(per1))
在前一篇文章已经分享过,当直接将实例对象转换为字符串类型,会默认调用__str__
方法。
但是如果是直接转为int
或float
或是bool
类型,系统会直接报错。
我们可以根据我们的需求重写方法。
一下是对应调用的方法:
类型转换 | 对应的方法 |
---|---|
int | __int__ |
float | __float__ |
str | __str__ |
bool | __bool__ |