PYTHON3中 类的继承(转帖)

原帖地址:https://www.cnblogs.com/lx3822/p/8808845.html

 

继承

1:什么是继承

继承是一种创建新类的方式,在python中,新建的类可以继承一个或多个父类,也就是说在python中支持一个儿子继承多个爹。

新建的类成为子类或者派生类。

父类又可以成为基类或者超类。

子类会遗传父类的属性。

2:为什么要用继承

减少代码冗余(也就是重复写代码)。

3:怎么用继承:

复制代码
我们定义两个类;
class parenclass1:
        pass

class parenclass2:
        pass


在定义两个类:

class subclass1: pass class subclass2: pass 我想让 : class parenclass1: 作为 class subclass1: 的父类。 pass pass 

应该这样用: class subclass1( parenclass1): 这就表示subclass1是子类,parenclass 是subclass1 的父类 pass
两个父类的话怎么表达呢?如下: class subclass2(parenclass1,parenclass2): pass 这就表示subclass2的父类是parenclass1,parenclass2 这两个
复制代码

 

想要查看子类的父类 应该怎样查看呢: 用__bases__  如下:

复制代码
class ParentClass1:
    pass

class ParentClass2:
    pass

class Subclass1(ParentClass1): pass class Subclass2(ParentClass1,ParentClass2): pass print(Subclass1.__bases__) #打印结果:(<class '__main__.ParentClass1'>,) print(Subclass2.__bases__) #打印结果: (<class '__main__.ParentClass1'>, <class '__main__.ParentClass2'>)
复制代码

经典类与新式类

复制代码
1、只有在python2中才分新式类和经典类,python3中统一都是新式类
2、在python2中 没有显示继承的object类的类,以及该类的子类都是经典类
3、在python2中,显示的声明继承object的类,以及该类的子类都是新式类 4、在python3中,无论是否继承object,都默认 继承object,即python3中所有类均为新式类 至于经典类 与新式类的区别,后面会有讨论。 提示:如果没有指定基类, python的类会默认继承object类, object是所有python类的基类。
复制代码

 

二、继承与抽象

继承描述的是子类与父类之间的关系,是一种什么的关系呢? 要找出这种关系, 必须先抽象在继承。

抽象即抽取类似或者说比较像的部分。

抽象分成两个层次:

1.将奥巴马和梅西这俩对象比较像的部分抽取成类; 

2.将人,猪,狗这三个类比较像的部分抽取成父类。

抽象最主要的作用是划分类别(可以隔离关注点,降低复杂度)

PYTHON3中 类的继承(转帖)_第1张图片

 

继承:是基于抽象的结果,通过编程语言去实现它,肯定是先经历抽象合格过程, 才能通过继承的方式去表达抽象的结构。

抽象只是分析和设计的过程,一个动作或者说一种技巧,通过抽象可以得到类

PYTHON3中 类的继承(转帖)_第2张图片

 

 例如:我们写一个老男孩的老师与学生的类,若是不涉及到继承的话  我们正常是这样写

复制代码
class OldboyTeacher:
    school = 'oldboy' def __init__(self,name,age,sex): self.name=name self.age=age self.sex=sex def change_score(self): print('teacher %s is changing score ' %self.name) class Oldboystudent:
    school = 'oldboy’
    def __init__(self,name,age,sex):
        self.name=name
        self.age=age
        self.sex=sex

    def choose(self):
        print('student %s choose course' %self.name) tea1 = OldboyTeacher('egon', 18, 'male') #OldboyTeacher.__init__(...) stu1=Oldboystudent('alex',73,'female') print(tea1.name,tea1.age,tea1.sex) # egon 18 male

print(stu1.name) #alex
复制代码

但是我们经过分析 发现里面里面有许多重复代码, 这时我们可以用到类的继承来写了。如下:

复制代码
class OldboyPeople:
    school ='oldboy' def __init__(self,name,age,sex): self.name=name self.age=age self.sex=sex class Oldboyteacher(OldboyPeople): def change_score(self): print('teacher %s is changing score ' %self.name)
class Oldboystudent(OldboyPeople): def choose(self): print('student %s choose course'%self.name) tea1 = Oldboyteacher('egon', 18, 'male') stu1=Oldboystudent('alex',73,'female')
print(tea1.name,tea1.age,tea1.sex)#egon 18 male print(stu1.name) #alex
复制代码

 三、基于继承在看属性查找

我们先看一个列子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class Foo:
     def f1(self):
         print( 'Foo.f1' )
 
     def f2(self): #self=obj
         print( 'Foo.f2' )    #在父类中找到发 f2属性,第3步打印这一行
         self.f1() #obj.f1()  第4步再去掉用self的f1属性
 
class Bar(Foo):
     def f1(self):#第五步, 在回到 object 自身的名称空间找f1属性,找到后调用
         print( 'Bar.f1' ) #第6步 执行
 
obj=Bar() #第一步 :类的实例化, 先得到一个空对象,
 
obj.f2()  #第2步:空对象调用f2属性 在自身寻找f2属性, 没有找到就去父类中寻找

#结果

Foo.f2
Bar.f1

注意子类的属性查找,一定是优先查找子类自己本身的属性与特征, 在本身没有的情况下 在去父类中查找。

这个还是非常有意思的,当执行obj.f2时,自己没有该属性,寻找父类属性,当找到f2属性时,执行self.f1,又优先取寻找自己的self.f1属性了,确实够贱。

self很重要,哪个self进去,属性都是先找self身上自己的。

四、派生

派生:子类定义自己新的属性,如果与父类同名,以子类自己的为准。

复制代码
class OldboyPeople:
    school = 'oldboy' def __init__(self, name, age, sex): self.name = name self.age = age self.sex = sex def f1(self): print('爹的f1') class OldboyTeacher(OldboyPeople): def change_score(self): print('teacher %s is changing score' %self.name) def f1(self): print('儿子的f1') tea1 = OldboyTeacher('egon', 18, 'male') tea1.f1()

#调用显示:儿子的f1
# 父类和子类中都有f1, 优先调用自己的属性,所以结果调用的是儿子的f1
复制代码

五、在子类中派生出的新方法重用父类的功能

拿上一案例来举例 在oldboyteacher 这个类中要添加薪水与级别。 然后调用。 有两种方式。

方式一:指名道姓的调用(与继承没有什么关系)

复制代码
class OldboyPeople:
    school ='oldboy' def __init__(self,name,age,sex): self.name=name self.age=age self.sex=sex def tell_info(self): print( ''' ====个人信息==== 姓名:%s 年龄:%s 性别:%s '''%(self.name,self.age,self.sex)) class OldboyTeacher(OldboyPeople): def __init__(self,name,age,sex,level,salary): OldboyPeople.__init__(self,name,age,sex) #在这里指明道姓来调用这一个函数里的属性 self.level =level self.salary=salary def tell_info(self): OldboyPeople.tell_info(self) #指名道姓的来调用这个函数里的属性 print(""" 等级:%s 薪资:%s """ %(self.level,self.salary))  tea1 = OldboyTeacher('egon', 18, 'male', 9, 3.1) print(tea1.name, tea1.age, tea1.sex, tea1.level, tea1.salary) tea1.tell_info() #打印结果: egon 18 male 9 3.1 ====个人信息==== 姓名:egon 年龄:18 性别:male 等级:9 薪资:3.1
复制代码

方法二、

用super()调用(严格依赖于继承)

super() 的返回值是一个特殊的对象,该对象专门用来调用父类中的属性, 一般在python2中,需要super(自己的类名,self),  而python3中,括号里面一般不填类名

复制代码
class OldboyPeople:
    school = 'oldboy' def __init__(self, name, age, sex): self.name = name self.age = age self.sex = sex def tell_info(self): print(""" ===========个人信息========== 姓名:%s 年龄:%s 性别:%s """ %(self.name,self.age,self.sex)) class OldboyTeacher(OldboyPeople): def __init__(self, name, age, sex, level, salary): super().__init__(name,age,sex) self.level = level self.salary = salary def tell_info(self): super().tell_info() print(""" 等级:%s 薪资:%s """ %(self.level,self.salary))  tea1 = OldboyTeacher('egon', 18, 'male', 9, 3.1) print(tea1.name, tea1.age, tea1.sex, tea1.level, tea1.salary) tea1.tell_info() #调用结果: egon 18 male 9 3.1 ===========个人信息========== 姓名:egon 年龄:18 性别:male 等级:9 薪资:3.1 
复制代码

 

六:经典类 与新式类

1新式类:

继承object的类,以及该类的子类,都是新式类

在python3中,如果一个类没有指定继承的父类,默认就继承object

所以说在python3中所有的类都是新式类

 

2经典类:(只有在python2才区分经典类和新式类):

没有继承object的类,以及该类的子类 都是经典类

1 经典类:深度优先

2 新式类:广度优先

如果继承关系为非菱形结果吗则会按照先找B 这一条分支,然后在找c这一条分支,最后找D这一条分支的顺序,直到找到我们想要的属性

 

当继承关系为菱形结构时

经典类查找顺序:

PYTHON3中 类的继承(转帖)_第3张图片

 

若是在A 类里自己没找到, 则会先去B类里去找, B类里没找到,就会在E类里找, 然后在G类里找,

G类里没找到 会去C 类里找, 然后去F 类里找,最后去D 类里找。

 

新式类查找顺序: 

PYTHON3中 类的继承(转帖)_第4张图片

按照图中1 ,2 , 3, 4,5,6的顺序查找, 这个为广度优先的查找方式

这个图 蛮好的,缺少了两个线,应该E(G)与F(G)也指向G(object),其实对于新式类的继承,基于广度与深度,我自己根本无法定论。

C3算法我没去仔细研究,但根据我自己的经验来看,会根据继承的类的顺序与深度相关。

下面上我自己写的测试代码:

class A(object):
    def ping(self):
        print('ping', self)


class B(A):
    def pong(self):
        print('pong', self)


class E(B):
    ...




class F(object):
    ...

class H(F):
    ...

class C(H):
    def pong(self):
        print('PONG', self)

class D(C, E):

    def ping(self):
        super(D, self).ping()
        print('post-ping', self)



if __name__ == '__main__':
    print(D.__mro__)

 

/usr/local/bin/python3.7 /Users/shijianzhong/study/Fluent_Python/第十二章/diamond.py
(, , , , , , , )

Process finished with exit code 0

 从运行结果来看,C线继承到object之前,一致沿着C走,后面再沿着E走。

object是两个父类的相交的基类结合点。

class A(object):
    def ping(self):
        print('ping', self)


class B(A):
    def pong(self):
        print('pong', self)


class E(B):
    ...




class F(object):
    ...

class H(A):
    ...

class C(H):
    def pong(self):
        print('PONG', self)

class D(C, E):

    def ping(self):
        super(D, self).ping()
        print('post-ping', self)



if __name__ == '__main__':
    print(D.__mro__)

 

/usr/local/bin/python3.7 /Users/shijianzhong/study/Fluent_Python/第十二章/diamond.py
(, , , , , , )

Process finished with exit code 0

 我把上面代码的H继承到了A,这样D的两个父类的基类相交点到A了,从继承来看,先寻找C线到A之前的类,然后再用E寻找到最后的object。

 

七:  super()依赖继承

super()会严格按照mro列表从当前查找到的位置继续往后查找

复制代码
class A:
    def test(self):
        print('A.test') #2 执行这一步 打印 super().f1 #3 然后在调用父类里的f1, 根据C.mro里的查找顺序执行到A 往后继续执行到B里去查找 class B: def f1(self): #4找到f1, 执行 print('from B') #5打印 class C(A,B): pass c=C() print(C.mro()) #调用属性的顺序 [, , , ] c.test() #1:C里没有 ,去A里调用

#打印结果

A.test

from B

复制代码

你可能感兴趣的:(PYTHON3中 类的继承(转帖))