Python进阶 - 特殊方法的使用

一、什么是特殊方法?

特殊方法可以理解为在调用内置方法时,程序背后真正被调用的方法,通常以双下划线开头和结尾,例如 __len__,有些地方也称其为"魔术方法"(Magic Method)。
比如,我们比较两个数的大小时,一般都会使用大于号或小于号进行判断:

# 命令行调用
>>> 1 < 2
True

实际上是在调用 __lt__() 方法:

# 命令行调用
>>> int.__lt__(1, 2)
True

二、特殊方法有什么用?

既然程序都已经自动调用了,那它有什么用呢?其实,它更多的用于我们自己编写的类中。下面我们自行编写一个向量类:

class Vector:
   def __init__(self, x=0, y=0):
       self.x = x  # x坐标
       self.y = y  # y坐标

我们在创建好两个向量实例后,发现单独显示的貌似时地址,将两个向量相加时也直接报错了,跟我们预想的完全不一样:

# 命令行调用
>>> v1 = Vector(1, 1)
>>> v1
<__main__.Vector object at 0x0000013C5E4CC9D0>
>>> v2 = Vector(2, 3)
>>> v2
<__main__.Vector object at 0x0000013C5E4CC8B0>
>>> v1 + v2
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unsupported operand type(s) for +: 'Vector' and 'Vector'
>>>

这时候特殊方法就派上用场了。下面通过完善这个向量类,介绍几款常用的特殊方法。 

三、几款常用的特殊方法

__repr__() 这个方法的主要作用是将一个对象用字符串的形式输出,便于我们进行辨认。
对于我们这个向量类,我们想让每个向量实例都输出成“Vector(m, n)”这样的格式,所以我们就可以使用 __repr__() 方法进行定义:

class Vector:
    def __init__(self, x=0, y=0): 
        self.x = x
        self.y = y
    def __repr__(self):
        # 定义Vector类的对象的字符串返回格式
        return 'Vector({}, {})'.format(self.x, self.y)

# 输出结果
>>> v1 = Vector(1, 1)
>>> v1
Vector(1, 1)

可以看到,输出的结果符合我们的预期。我们继续通过几个特殊方法来完成向量的模、加法以及数乘等操作:

from math import hypot

class Vector:
    def __init__(self, x=0, y=0): 
        self.x = x
        self.y = y
    def __repr__(self):
        # 定义Vector类的对象的字符串返回格式
        return 'Vector({}, {})'.format(self.x, self.y)
    def __abs__(self):
        # 通过计算x,y坐标的欧几里得距离,返回向量的模
        return hypot(self.x, self.y)
    def __bool__(self):
        # 判断向量的模是否为0,从而进行布尔运算
        return bool(abs(self))
    def __add__(self, other):
        # 定义两个向量的加法
        x = self.x + other.x
        y = self.y + other.y
        return Vector(x, y)
    def __mul__(self, scalar):
        # 定义一个数scalar与向量的乘积
        return Vector(self.x * scalar, self.y * scalar)

# 命令行调用
>>> v1 = Vector(1, 2)    # 定义向量v1的坐标为(1, 2)
>>> v2 = Vector(2, 2)    # 定义向量v2的坐标为(2, 2)
>>> v1
Vector(1, 2)
>>> v2
Vector(2, 2)
>>> v1 + v2    # 向量相加
Vector(3, 4)
>>> v1 * 3      # 向量的数乘
Vector(3, 6)
>>> abs(v1 + v2)    # 求 v1 + v2 的模
5.0
>>> bool(v1)    # 对向量v1进行布尔运算True
>>> bool(Vector(0, 0))    # 对向量(0, 0)进行布尔运算
False

通过对一系列特殊方法进行定义,我们基本上完成了向量的基本操作。这让我们这个Vector类变得生动起来。 

四、特殊方法的意义

通过在自定义的类中重写一些原生类方法背后的特殊方法,可以使得自定义的类也拥有和原生类相同的访问、操作方式,让使用者尽量不会感受到自定义类与原生类的差异,从而保持语言上的一致性。

五、如何查找特殊方法?

通过Python语言参考手册的“Data Model”,可以查找到83个特殊方法的名字,其中47个用于实现算术运算、位运算和比较操作。



注:文章内容出自《流畅的Python》第一章

posted @ 2021-06-27 13:22  dev-liu  阅读(177)  评论(0编辑  收藏  举报