有小伙伴会问,什么是python的魔法方法,python的魔法方法有什么用呢,
它们在面向对象的Python的处处皆是。它们是一些可以让你对类添加“魔法”的特殊方法。 它们经常是两个下划线包围来命名的。
我感觉魔法方法很牛逼的,
目前我们常见的魔法方法大致可分为以下几类:
- 构造与初始化
- 类的表示
- 访问控制
- 比较操作
- 容器类操作
- 可调用对象
- Pickling序列化
在我们写python中最常见的就是__init__,这是python的构造方法,这是初始化对象,定义一个对象的初始化的值,__new__ 这个我们在实际的很少用到吧, 不过在一些开源的框架到我们能用到这个,__new__返回值是一个实例,
__del__
这个方法代表析构方法,也就是在对象被垃圾回收时被调用。但是请注意,执行del x
不一定会执行此方法
__str__
/__repr__
这两个魔法方法一般会放到一起进行讲解,它们的主要差别为:
-
__str__
强调可读性,而__repr__
强调准确性/标准性 -
__str__
的目标人群是用户,而__repr__
的目标人群是机器,它的结果是可以被执行的 -
%s
调用__str__
方法,而%r
调用__repr__
方法
__setattr__
通过此方法,对象可在在对属性进行赋值时进行控制,所有的属性赋值都会经过它。
一般常用于对某些属性赋值的检查校验逻辑,例如age
不能小于0,否则认为是非法数据等等。
__getattr__
很多同学以为此方法是和__setattr__
完全对立的,其实不然!
这个方法只有在访问某个不存在的属性时才会被调用,看上面的例子,由于gender
属性在赋值时,忽略了此字段的赋值操作,所以此属性是没有被成功赋值给对象的。当访问这个属性时,__getattr__
被调用,返回unknown
。
__del__
删除对象的某个属性时,此方法被调用。一般常用于某个属性必须存在,否则无法进行后续的逻辑操作,会重写此方法,对删除属性逻辑进行检查和校验。
__getattribute__
这个方法我们很少用到,它与__getattr__
很容易混淆。它与前者的区别在于:
-
__getattr__
访问某个不存在的属性被调用,__getattribute__
访问任意属性被调用
__getattr__
只针对属性访问,__getattribute__
不仅针对所有属性访问,还包括方法调用
__iter__
执行for x in obj
时触发执行,用于迭代容器内的元素。
__contains__
执行x in obj
时触发执行,用于判断某个元素是否存在于容器中。
__reversed__
执行reversed(obj)
时触发执行,用于反转容器的元素,具体的反转逻辑可自己实现。
反射
__nstancecheck__(self, instance)
:检查一个实例是否是你定义类中的一个实例
__ubclasscheck__(self, subclass)
:检查一个类是否是你定义类的子类
数值操作符
就像你可以使用比较操作符来比较类的实例,你也可以定义数值操作符的行为。固定好你的安全带,这样的操作符真的有很多。看在组织的份上,我把它们分成了五类:一元操作符,常见算数操作符,反射算数操作符,增强赋值操作符,和类型转换操作符。
一元操作符
一元操作符只有一个操作符。
- __pos__(self)
实现取正操作,例如 +some_object。
-
__neg__(self)
实现取负操作,例如 -some_object。
-
__abs__(self)
实现内建绝对值函数 abs() 操作。
-
__invert__(self)
实现取反操作符 ~。
-
__round__(self, n)
实现内建函数 round() ,n 是近似小数点的位数。
-
__floor__(self)
实现 math.floor() 函数,即向下取整。
-
__ceil__(self)
实现 math.ceil() 函数,即向上取整。
-
__trunc__(self)
实现 math.trunc() 函数,即距离零最近的整数。
常见算数操作符
现在,我们来看看常见的二元操作符(和一些函数),像+,-,*之类的,它们很容易从字面意思理解。
-
__add__(self, other)
实现加法操作。
-
__sub__(self, other)
实现减法操作。
-
__mul__(self, other)
实现乘法操作。
-
__floordiv__(self, other)
实现使用 // 操作符的整数除法。
-
__div__(self, other)
实现使用 / 操作符的除法。
-
__truediv__(self, other)
实现 _true_ 除法,这个函数只有使用 from __future__ import division 时才有作用。
-
__mod__(self, other)
实现 % 取余操作。
-
__divmod__(self, other)
实现 divmod 内建函数。
-
__pow__
实现 ** 操作符。
-
__lshift__(self, other)
实现左移位运算符 << 。
-
__rshift__(self, other)
实现右移位运算符 >> 。
-
__and__(self, other)
实现按位与运算符 & 。
-
__or__(self, other)
实现按位或运算符 | 。
-
__xor__(self, other)
实现按位异或运算符 ^ 。
class DictDemo: def __init__(self,key,value): self.dict={} self.dict[key]=value def __getitem__(self,key): return self.dict[key] def __setitem__(self,key,value): self.dict[key]="key's value is %s"%value def __len__(self): return (len(self.dict)) dictDemo=DictDemo('key0','value1') print(dictDemo['key0']) dictDemo['key1']='beijing' print(dictDemo['key1']) print(len(dictDemo)) class word(str): def __eq__(self,other): return len(self)==len(other) w1=word('idshfisdojf') w2=word('sadsadas') print(w1==w2) print(len(w1)) class Apple(object): def __init__(self,name,size): self.name=name self.size=size def __str__(self): return 'name is {}: size is {}'.format(self.name,self.size) print(Apple('liwanlei','100'))
class Person(object): def __init__(self,name,age): self.name=name self.age=age def __str__(self): return 'name :%s, age:%s'%(self.name,self.age) def __repr__(self): return "Person( '%s'%s)"%(self.name,self.age) person=Person('leizi','18') print(str(person))#name :leizi, age:18 print(repr(person))#Person( 'leizi'18)