课时41:魔法方法:构造和析构

目录:

  一、_ _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课后习题及答案

*******************************

 

posted @ 2018-08-24 15:29  那是个好男孩  阅读(295)  评论(0编辑  收藏