python入门

python第41课练习—魔法方法:构造和析造

2019-06-15  本文已影响0人  YoYoYoo

1、是哪个特征让我们一眼就认出这货是魔法方法?

答:魔法方法总是被双下划线包围,例如__init__。

2、类实例化对象所调用的第一个方法是什么?

答:__new__是在一个对象实例化的时候所调用的第一个方法。它跟其他的魔法方法不同,它的第一个参数不是self,而是这个类(cls),而其他的参数会直接传递给__init__方法的。

3、什么时候我们需要在类中明确写出__init__方法?

答:当我们的实例对象需要有明确的初始化步骤的时候,你可以在__init__方法中部署初始化的代码。
举个例子:

# 我们定义一个矩形类,需要长和宽两个参数,拥有计算周长和面积两个方法
# 我们需要在对象初始化的时候拥有“长”和“宽”两个参数,因此我们需要重写__init__方法
# 因为我们说过,__init__方法是类在实例化成对象的时候首先会调用的一个方法,大家可以理解吗?

class Rectangle:
    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

rect = Rectangle(3,4)
a = rect.getPeri()
b = rect.getArea()
print(a)
print(b)

>>>14
>>>12

4、请问下边代码存在什么问题?

class Test:
    def __init__(self,x,y):
        return x + y

答:编程中需要注意到__init__方法的返回值一定是None,不能是其他!
如:

>>> t = Test(3,4)
Traceback (most recent call last):
  File "<pyshell#3>", line 1, in <module>
    t = Test(3,4)
TypeError: __init__() should return None, not 'int'

5、请问__new__方法是负责什么任务?

答:__new__方法主要是在一个对象实例化时返回一个实例对象,通常是参数cls这个类的实例化对象,当然你也可以返回其他对象。

6、__del__魔法方法什么时候会被自动调用?

答:如果说__init__和__new__方法是对象的构造器的话,那么Python也提供了一个析构器,叫做__del__方法。当对象将要被销毁的时候,这个方法就会被调用。
但一定要注意的是,并非del x 就相当于自动调用x.__del__(),__del__方法是当垃圾回收机制回收这个对象的时候调用的。

练习

1、小李做事常常丢三落四的,写代码也一样,常常打开了文件又忘记关闭。你能不能写一个FileObject类,给文件对象进行包装,从而确认在删除对象时文件能自动关闭?

答:只要做到灵活搭配__init__和__del__魔法方法,即可做到收放自如。
代码清单:

class FileObject:
    '''给文件对象进行包装从而确认在删除时文件流关闭'''

    def __init__(self,filename = 'sample.txt'):
        # 读写模式打开一个文件
        self.new_file = open(filename,'r+')

    def __def__(self):
        self.new_file.close()
        del self.new_file

2、按照以下要求,定义一个类实现摄氏度到华氏度的转化(转化公式:华氏度 = 摄氏度*1.8 + 32)

要求:我们希望这个类尽量简练的实现功能,如下:

>>>print(C2F(32))
89.6

答:为了尽量简练的实现功能,我们采取了“偷龙转凤”的小技巧,在类进行初始化之前,通过“掉包”arg参数,让实例对象直接返回计算后的结果。
代码清单:

class C2F(float):
    "摄氏度转换为华氏度"
    def __new__(cls,arg = 0.0):
        return float.__new__(cls,arg * 1.8 + 32)

3、定义一个类继承于int类型,并实现一个特殊功能:当传入的参数是字符串的时候,返回该字符串中所有字符的ASCⅡ码的和(使用ord()获得一个字符的ASCⅡ码值)

实现如下:

>>> print(Nint(123))
123
>>> print(Nint(1.5))
1
>>> print(Nint('A'))
65
>>> print(Nint('FishC'))
461

代码清单:

class Nint(int):
    def __new__(cls,arg = 0):
        if isinstance(arg,str):
            total = 0
            for each in arg:
                total += ord(each)
            arg = total
        return int.__new__(cls,arg)
上一篇下一篇

猜你喜欢

热点阅读