魔法方法总是被双下划线包围,例如:__init__()
魔法方法是面向对象的python的一切,它的魔力体现在总能在合适的时候调用。
先来介绍析构和构造的三个魔法方法:
__init__():构造方法,类在实例化成对象的时候会首先调用这个方法(可选,根据需求)
__new__():才是在一个对象实例化的时候所调用的一个方法,与其他魔方方法不同,它的第一个参数是cls(类),其他参数会传给__init__()
__del__():如果说__init__()和__new__()是对象的构造器话,__del__()方法将是打工对象被销毁时调用
1、__init__(self[,...])
实例:
1 class Rectangle: 2 ''' 3 定义个一个矩形列,需要长宽两个参数 4 需要对象在初始化的时候拥有长宽两个参数,因此需要重写__init__()方法 5 ''' 6 def __init__(self,x,y): #__init__()返回值一定是None,所以不需要return语句 7 self.x = x 8 self.y = y 9 def getPerl(self): 10 return (self.x + self.y) * 2 11 def getArea(self): 12 return self.x * self.y 13 14 rect = Rectangle(3,4) 15 print(rect.getArea(),rect.getPerl())
打印结果:
14
12
这里需要注意的是,__init__()方法的返回值一定是None,不能是其他,否则将报错:
1 class A(): 2 def __init__(self): 3 return "A" 4 5 cup = A()
执行结果:
Traceback (most recent call last): File "<pyshell#26>", line 1, in <module> cup = A() TypeError: __init__() should return None, not 'str'
所以一般在需要进行初始化的时候才重写__init__()方法,其实__init__()并不是实例化对象时第一个被调用的魔法方法
2、__new__(cls[,...])
__new__()才是在实例化对象时所调用的第一个方法。它跟其他魔法方法不同,它的第一个参数不适合self而是类(cls),而其他的参数会直接传递给__init__()方法
__new__()方法需要返回一个实例化对象,通常是cls这个实例化的对象,当然也可以返回其他对象。
__new__()方法平时很少去重写它,一般让python用默认的方案执行就可以了。但是有一种情况需要重写这个魔法方法,就是当继承一个不可变类型的时候,它的特性就尤为重要了。
1 class CapStr(str): 2 def __new__(cls,string): 3 string = string.upper() 4 return str.__new__(str,string) #这里一定是返回继承类的__new__()方法,否则容易造成死循环 5 6 capstr = CapStr('good afternoon') 7 print(capstr) 8 9 #结果为:GOOD AFTERNOON
这里返回的str.__new__(str,string)做法是很值得推荐的,只需要重写关组的内容,其他的琐碎东西交给python的默认机制去完成就可以了,毕竟python本身比我们自己写的要好很多
3、__del__(self)
如果说__init__()和__new__()方法是对象的构造器的话,那么python也析构了一个构造器,__del__()方法,当对象被销毁时,这个方法就一定会被调用,但要注意的是,并非del x就等同于自动调用了x.__del__(),__del__()方法是当垃圾回收机制回收这个对象的时候调用的
>>> class C(): def __init__(self): print('我是__init__()方法') def __del__(self): print('我是__del__()方法') >>> c1 = C() 我是__init__()方法 >>> c2 = c1 >>> c3 = c2 >>> del c1 >>> del c2 >>> del c3 我是__del__()方法