python类继承的重写和super

给已经存在的类添加新的行为,继承是非常好的实现方式。但是如果要改变行为呢?比如在Python继承扩展内置类,我们的contact类只允许一个名字和一个邮箱,但是如果要对某些人增加电话号码呢?这里可以通过非常简单的给他增加一个电话号码的属性来实现,但是如果想这第3个变量在初始化类的时候就可用,就需要重写__init__函数

重写就是在子类里用一个(和超类相同的名字)新的方法来改变或者覆盖超类里的这个方法。做这个不需要特殊的语法,会自动执行子类新创建的方法而不是超类的方法。以下是在之前Contact类中,使用Friend子类来继承

class ContactList(list):
    def search(self, name):
        '''Return all contacts that contain the search value
           in their name.'''
        matching_contacts = []
        for contact in self:
            if name in contact.name:
                matching_contacts.append(contact)
        return matching_contacts

class Contact:
    all_contacts = ContactList()
  
    def __init__(self, name, email):
        self.name = name
        self.email = email
        Contact.all_contacts.append(self)


class Supplier(Contact):
    def order(self, order):
        print("If this were a real system we would send"
                 "{} order to {}".format(order, self.name))


class Friend(Contact):
    '''以下是重写方法,注意这个方法没有将name和email等加入父类的all_contact   '''
    def __init__(self, name, email, phone):
        self.name = name 
        self.email = email
        self.phone = phone

class Friend1(Contact):
    '''通过super得到父类对象的实例,并且调用这个对象的__init__方法,
       传递给它预期的参数,然后这个类做了自己的初始化,即设置phone属性'''
    def __init__(self, name, email, phone):
        super().__init__(name, email)
        self.phone = phone

不只有__init__,任何方法都可以被重写。在这个例子中,Contact和Friend类有重复的代码来创建name和email,这会导致维护变得复杂。需要警惕的是,Friend类忽略了把自己加到all_contacts列表里,这个列表是我们在Contact类里创建的。将以上代码保存为order.py脚本,如下是在交互的解释器里运行的结果如下:

$ python -i order.py 
>>> c1 = Contact("John A", "johna@example.com")
>>> c2 = Contact("John B", "johnb@example.com")
>>> c3 = Contact("Jenna C", "jennac@example.com")
>>> c4 = Friend("Friend c", "Friendc@example.com",'0592-5566778')
>>> c5 = Friend("Friend 5", "Friend5@example.com",'0592-1122334')
>>> c6 = Friend1("Friend 6", "Friend6@example.com",'0592-4455667')
>>> c7 = Friend1("Friend 7", "Friend7@example.com",'0592-7788990')
>>> [c.name for c in Contact.all_contacts]
['John A', 'John B', 'Jenna C', 'Friend 6', 'Friend 7']

我们真正需要的是一种可以调用父类的代码的方法,这个就是super函数的功能,他返回一个父类的实例化对象,允许我们直接调用父类的方法,如上例中的Friend1类。

super()可以在任何方法中调用,不只是__init__方法,这就意味着通过重写和调用super,可以修改所有的方法。可以在方法的任何位置用super。

 

参考:

1、《Python3 面向对象编程》 [加]Dusty Philips 著

posted @ 2017-12-29 08:56  anovana  阅读(6707)  评论(0编辑  收藏  举报