在Python中以两个下划线开头的方法,__init__、__str__、__doc__、__new__等,被称为"魔术方法"(Magic methods)。魔术方法在类或对象的某些事件出发后会自动执行,如果希望根据自己的程序定制自己特殊功能的类,那么就需要对这些方法进行重写。
注意:Python 将所有以 __(两个下划线)开头的类方法保留为魔术方法。所以在定义类方法时,除了上述魔术方法,建议不要以 __ 为前缀。
魔术方法列表
魔法方法 | 含义 | |
---|---|---|
基本的魔法方法 | ||
__new__(cls[, ...]) | __new__ 是在一个对象实例化的时候所调用的第一个方法。它的第一个参数是这个类,其他的参数是用来直接传递给 __init__ 方法。__new__ 决定是否要使用该 __init__ 方法,因为 __new__ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 __new__ 没有返回实例对象,则 __init__ 不会被调用。 __new__ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string | |
__init__(self[, ...]) | 构造器,当一个实例被创建的时候调用的初始化方法 | |
__del__(self) | 析构器,当一个实例被销毁的时候调用的方法 | |
__call__(self[, args...]) | 允许一个类的实例像函数一样被调用:x(a, b) 调用 x.__call__(a, b) | |
__len__(self) | 定义当被 len() 调用时的行为 | |
__repr__(self) | 定义当被 repr() 调用或者直接执行对象时的行为 | |
__str__(self) | 定义当被 str() 调用或者打印对象时的行为 | |
__bytes__(self) | 定义当被 bytes() 调用时的行为 | |
__hash__(self) | 定义当被 hash() 调用时的行为 | |
__bool__(self) | 定义当被 bool() 调用时的行为,应该返回 True 或 False | |
__format__(self, format_spec) | 定义当被 format() 调用时的行为 | |
有关属性 | ||
__getattr__(self, name) | 定义当用户试图获取一个不存在的属性时的行为 | |
__getattribute__(self, name) | 定义当该类的属性被访问时的行为 | |
__setattr__(self, name, value) | 定义当一个属性被设置时的行为 | |
__delattr__(self, name) | 定义当一个属性被删除时的行为 | |
__dir__(self) | 定义当 dir() 被调用时的行为 | |
__get__(self, instance, owner) | 定义当描述符的值被取得时的行为 | |
__set__(self, instance, value) | 定义当描述符的值被改变时的行为 | |
__delete__(self, instance) | 定义当描述符的值被删除时的行为 | |
比较操作符 | ||
__lt__(self, other) | 定义小于号的行为:x < y 调用 x.__lt__(y) | |
__le__(self, other) | 定义小于等于号的行为:x <= y 调用 x.__le__(y) | |
__eq__(self, other) | 定义等于号的行为:x == y 调用 x.__eq__(y) | |
__ne__(self, other) | 定义不等号的行为:x != y 调用 x.__ne__(y) | |
__gt__(self, other) | 定义大于号的行为:x > y 调用 x.__gt__(y) | |
__ge__(self, other) | 定义大于等于号的行为:x >= y 调用 x.__ge__(y) | |
算数运算符 | ||
__add__(self, other) | 定义加法的行为:+ | |
__sub__(self, other) | 定义减法的行为:- | |
__mul__(self, other) | 定义乘法的行为:* | |
__truediv__(self, other) | 定义真除法的行为:/ | |
__floordiv__(self, other) | 定义整数除法的行为:// | |
__mod__(self, other) | 定义取模算法的行为:% | |
__divmod__(self, other) | 定义当被 divmod() 调用时的行为 | |
__pow__(self, other[, modulo]) | 定义当被 power() 调用或 ** 运算时的行为 | |
__lshift__(self, other) | 定义按位左移位的行为:<< | |
__rshift__(self, other) | 定义按位右移位的行为:>> | |
__and__(self, other) | 定义按位与操作的行为:& | |
__xor__(self, other) | 定义按位异或操作的行为:^ | |
__or__(self, other) | 定义按位或操作的行为:\ | |
反运算 | ||
__radd__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rsub__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rmul__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rtruediv__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rfloordiv__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rmod__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rdivmod__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rpow__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rlshift__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rrshift__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rand__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__rxor__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
__ror__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) | |
增量赋值运算 | ||
__iadd__(self, other) | 定义赋值加法的行为:+= | |
__isub__(self, other) | 定义赋值减法的行为:-= | |
__imul__(self, other) | 定义赋值乘法的行为:*= | |
__itruediv__(self, other) | 定义赋值真除法的行为:/= | |
__ifloordiv__(self, other) | 定义赋值整数除法的行为://= | |
__imod__(self, other) | 定义赋值取模算法的行为:%= | |
__ipow__(self, other[, modulo]) | 定义赋值幂运算的行为:**= | |
__ilshift__(self, other) | 定义赋值按位左移位的行为:<<= | |
__irshift__(self, other) | 定义赋值按位右移位的行为:>>= | |
__iand__(self, other) | 定义赋值按位与操作的行为:&= | |
__ixor__(self, other) | 定义赋值按位异或操作的行为:^= | |
__ior__(self, other) | 定义赋值按位或操作的行为: | = |
一元操作符 | ||
__pos__(self) | 定义正号的行为:+x | |
__neg__(self) | 定义负号的行为:-x | |
__abs__(self) | 定义当被 abs() 调用时的行为 | |
__invert__(self) | 定义按位求反的行为:~x | |
类型转换 | ||
__complex__(self) | 定义当被 complex() 调用时的行为(需要返回恰当的值) | |
__int__(self) | 定义当被 int() 调用时的行为(需要返回恰当的值) | |
__float__(self) | 定义当被 float() 调用时的行为(需要返回恰当的值) | |
__round__(self[, n]) | 定义当被 round() 调用时的行为(需要返回恰当的值) | |
__index__(self) | 1. 当对象是被应用在切片表达式中时,实现整形强制转换 2. 如果你定义了一个可能在切片时用到的定制的数值型,你应该定义 __index__ 3. 如果 __index__ 被定义,则 __int__ 也需要被定义,且返回相同的值 | |
上下文管理(with 语句) | ||
__enter__(self) | 1. 定义当使用 with 语句时的初始化行为2. __enter__ 的返回值被 with 语句的目标或者 as 后的名字绑定 | |
__exit__(self, exc_type, exc_value, traceback) | 1. 定义当一个代码块被执行或者终止后上下文管理器应该做什么2. 一般被用来处理异常,清除工作或者做一些代码块执行完毕之后的日常工作 | |
容器类型 | ||
__len__(self) | 定义当被 len() 调用时的行为(返回容器中元素的个数) | |
__getitem__(self, key) | 定义获取容器中指定元素的行为,相当于 self[key] | |
__setitem__(self, key, value) | 定义设置容器中指定元素的行为,相当于 self[key] = value | |
__delitem__(self, key) | 定义删除容器中指定元素的行为,相当于 del self[key] | |
__iter__(self) | 定义当迭代容器中的元素的行为 | |
__reversed__(self) | 定义当被 reversed() 调用时的行为 | |
__contains__(self, item) | 定义当使用成员测试运算符(in 或 not in)时的行为 |
魔术方法应用之对象使用集合去重
列表 / 集合 / 字典 - 底层实现原理
列表 - 双向循环链表
集合 / 字典 - 哈希存储 - 存取数据效率高 - O(1)
如果自定义对象要放到set中那么需要重写两个魔术方法
- __hash__ - 计算出对象对应的哈希码
- __eq__ - ==运算符对应的方法用来比较两个对象是否相同
如果两个对象eq的结果为True那么必须有相同的哈希码
如果两个对象哈希码相同但是eq的结果可能是True也可能是False
相同对象必须有相同的哈希码,不同对象可能会产生相同的哈希码(碰撞)
class Student(object):
def __init__(self, stuid, name, gender):
self.stuid = stuid
self.name = name
self.gender = gender
def __hash__(self):
return self.stuid
def __eq__(self, other):
return self.stuid == other.stuid and \
self.name == other.name
def main():
set1 = {
Student(1001, '骆昊', True),
Student(1001, '骆昊', False),
Student(1001, '王大锤', True)
}
print(len(set1))
print(set1)
if __name__ == '__main__':
main()
魔术方法应用之对象使用上下文语法
class Student(object):
def __init__(self, stuid, name, gender):
self.stuid = stuid
self.name = name
self.gender = gender
def __enter__(self):
return self
def __exit__(self, exception_type, exception_value, traceback):
pass
def __repr__(self):
return f'{self.stuid} - {self.name} - {self.gender}'
def main():
with Student(1003, '李白菜', False) as stu:
print(stu.name)
if __name__ == '__main__':
main()