目录:
一、_ _init_ _(self[, ...])
二、_ _new_ _(cls[, ...])
三、_ _del_ _(self)
四、课时41课后习题及答案
说的那么厉害,那什么是魔法方法呢?
(1)魔法方法总是被双下划线包围,例如_ _init_ _()。
(2)魔法方法是面对对象的Python的一切。
(3)魔法方法的“魔力”体现在它们总能够在适当的时候被调用。
**************************
一、_ _init_ _(self[, ...])
**************************
之前我们讨论过 _ _init_ _()方法,说它相当于其它面向对象编程语言的构造方法,也就是类在实例化成对象的时候首先会调用的一个方法。
也许你会问:“有些时候在定义时写_ _init_ _()方法,有些时候却没有,这是为什么呢?”举个例子:
#p12_1.py
class Rectangle:
"""
定义一个矩形类,
需要长和宽两个数据,
拥有计算周长和面积的两个办法。
拥有对象在初始化的时候拥有"长"和"宽"两个参数,
因此需要重写_ _init_ _()方法,因为我们说过,
_ _init_ _()方法是类在实例化成对象的时候首先会调用的一个方法,
"""
def __init__(self,x,y):
self.x = x
self.y = y
def getPeri(self):
return (self.x + self.y) * 2
def getArea(self):
return self.x * self.y
>>> #先运行p12_1.py
>>> rect = Rectangle(3,4)
>>> rect.getPeri()
14
>>> rect.getArea()
12
这里需要注意的是,_ _init_ _()方法的返回值一定是None,不能是其它:
>>> class A:
def __init__(self):
return "A for A - Cup"
>>> cup = A()
Traceback (most recent call last):
File "<pyshell#4>", line 1, in <module>
cup = A()
TypeError: __init__() should return None, not 'str'
所以一般在需要进行初始化的时候才重写__init__()方法。其实,这个__init__()并不是实例化对象时第一个被调用的魔法方法。
***************************
二、_ _new_ _(cls[, ...])
***************************
_ _new_ _()才是在一个对象实例化的时候所调用的第一个方法。它跟其它魔法方法不同,它的第一个参数不是self而时这个类(cls),而其它参数会直接传递给_ _init_ _()方法的。
_ _new_ _()方法需要返回一个实例对象,通常是cls这个类实例化的对象,当然你也可以返回其它对象。
_ _new_ _()方法平时很少去重写它,一般让Python用默认的方案执行就可以了。但是又一种情况需要重写这个魔法方法,就是当继承一个不可变的类型的时候,它的特性就显得尤为重要了。
>>> class CapStr(str):
def __new__(cls,string):
string = string.upper()
return str.__new__(cls,string)
>>> a = CapStr("I love ZWW")
>>> a
'I LOVE ZWW'
这里返回str.__new__(cls,string)这种做法是值得推崇的,只需要重写我们关注的那部分内容,然后其它的琐碎东西交给Python的默认机制去完成就可以了,毕竟它们出错的几率要比我们自己写小很多。
*********************
三、_ _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__()方法,我被调用了...
*******************************
四、课时41课后习题及答案
*******************************