什么是运算符重载,Python可重载运算符有哪些?

前面章节介绍了 Python 中的各个序列类型,每个类型都有其独特的操作方法,例如列表类型支持直接做加法操作实现添加元素的功能,字符串类型支持直接做加法实现字符串的拼接功能,也就是说,同样的运算符对于不同序列类型的意义是不一样的,这是怎么做到的呢?

其实在 Python 内部,每种序列类型都是 Python 的一个类,例如列表是 list 类,字典是 dict 类等,这些序列类的内部使用了一个叫作“重载运算符”的技术来实现不同运算符所对应的操作。

所谓重载运算符,指的是在类中定义并实现一个与运算符对应的处理方法,这样当类对象在进行运算符操作时,系统就会调用类中相应的方法来处理。

这里给大家举一个与重载运算符相关的实例:

class MyClass: #自定义一个类
    def __init__(self, name , age): #定义该类的初始化函数
        self.name = name #将传入的参数值赋值给成员交量
        self.age = age
    def __str__(self): #用于将值转化为字符串形式,等同于 str(obj)
        return "name:"+self.name+";age:"+str(self.age)
   
    __repr__ = __str__ #转化为供解释器读取的形式
   
    def __lt__(self, record): #重载 self<record 运算符
        if self.age < record.age:
            return True
        else:
            return False
   
    def __add__(self, record): #重载 + 号运算符
        return MyClass(self.name, self.age+record.age)

myc = MyClass("Anna", 42) #实例化一个对象 Anna,并为其初始化
mycl = MyClass("Gary", 23) #实例化一个对象 Gary,并为其初始化
print(repr(myc)) #格式化对象 myc,
print(myc) #解释器读取对象 myc,调用 repr
print (str (myc)) #格式化对象 myc ,输出"name:Anna;age:42"
print(myc < mycl) #比较 myc<mycl 的结果,输出 False
print (myc+mycl) #进行两个 MyClass 对象的相加运算,输出 "name:Anna;age:65"

输出结果为:

name:Anna;age:42
name:Anna;age:42
name:Anna;age:42
False
name:Anna;age:65

这个例子中,MyClass 类中重载了 repr、str、<、+ 运算符,并用 MyClass 实例化了两个对象 myc 和 mycl。

通过将 myc 进行 repr、str 运算,从输出结果中可以看到,程序调用了重载的操作符方法 __repr__ 和 __str__。而令 myc 和 mycl 进行 < 号的比较运算以及加法运算,从输出结果中可以看出,程序调用了重载 < 号的方法 __lt__ 和 __add__ 方法。

那么,Python 类支持对哪些方法进行重载呢?这个给大家提供一个表格(表 1),列出了 Python 中常用的可重载的运算符,以及各自的含义。

表 1 Python 常用重载运算符
重载运算符含义
__new__创建类,在 __init__ 之前创建对象
__init__类的构造函数,其功能是创建类对象时做初始化工作。
__del__ 析构函数,其功能是销毁对象时进行回收资源的操作
__add__加法运算符 +,当类对象 X 做例如 X+Y 或者 X+=Y 等操作,内部会调用此方法。但如果类中对 __iadd__ 方法进行了重载,则类对象 X 在做 X+=Y 类似操作时,会优先选择调用 __iadd__ 方法。
__radd__当类对象 X 做类似 Y+X 的运算时,会调用此方法。
__iadd__重载 += 运算符,也就是说,当类对象 X 做类似 X+=Y 的操作时,会调用此方法。
__or__“或”运算符 |,如果没有重载 __ior__,则在类似 X|Y、X|=Y 这样的语句中,“或”符号生效
__repr__,__str__格式转换方法,分别对应函数 repr(X)、str(X)
__call__函数调用,类似于 X(*args, **kwargs) 语句
__getattr__点号运算,用来获取类属性
__setattr__属性赋值语句,类似于 X.any=value
__delattr__删除属性,类似于 del X.any
__getattribute__获取属性,类似于 X.any
__getitem__索引运算,类似于 X[key],X[i:j]
__setitem__索引赋值语句,类似于 X[key], X[i:j]=sequence
__delitem__ 索引和分片删除
__get__, __set__, __delete__描述符属性,类似于 X.attr,X.attr=value,del X.attr
__len__ 计算长度,类似于 len(X)
__lt__,__gt__,__le__,__ge__,__eq__,__ne__ 比较,分别对应于 <、>、<=、>=、=、!= 运算符。
__iter__,__next__迭代环境下,生成迭代器与取下一条,类似于 I=iter(X) 和 next()
__contains__成员关系测试,类似于 item in X
__index__ 整数值,类似于 hex(X),bin(X),oct(X)
__enter__,__exit__在对类对象执行类似 with obj as var 的操作之前,会先调用 __enter__ 方法,其结果会传给 var;在最终结束该操作之前,会调用 __exit__ 方法(常用于做一些清理、扫尾的工作)