博客
关于我
Python菱形继承的初始化问题和继承顺序
阅读量:567 次
发布时间:2019-03-09

本文共 3528 字,大约阅读时间需要 11 分钟。

Python菱形继承的初始化问题和继承顺序

Python中,类通过继承的方式,子类可以获得父类的非私有属性和非私有方法,不需要自己再重新实现。

继承可以多层继承,即可以多代继承。也可以多继承,即一个子类可以继承多个父类。

参考:

一、菱形继承简介

在多层继承和多继承同时使用的情况下,就会出现复杂的继承关系,多重多继承。

其中,就会出现菱形继承。如下图所示。

菱形继承也叫钻石继承,在这种结构中,当D类的对象使用一个属性时,首先会在D类中查找是否有该属性和方法,如果没有则会到父类中查找,如果还没有则会继续往父类的父类中查找。

根据这种关系,如果D类中没有找到属性和方法,是去B类中查找还是去C类中查找?

假设是去B类中查找,如果B类中也没有找到,下一步是去C类中还是A类中呢?

这些问题都是菱形继承所带来的问题,接下来我们来看Python中是怎么处理的。

二、菱形继承的初始化问题

class Electrical(object):    def __init__(self, name):        self.name = name        print('Electrical init')class Phone(Electrical):    def __init__(self, name, price):        Electrical.__init__(self, name)        self.price = price        print('Phone init')class Computer(Electrical):    def __init__(self, name, config):        Electrical.__init__(self, name)        self.config = config        print('Computer init')class HuaWei(Phone, Computer):    def __init__(self, name, price, config):        Phone.__init__(self, name, price)        Computer.__init__(self, name, config)        print('HuaWei init')h = HuaWei('huawei', 100, 'i7')

运行结果:

Electrical initPhone initElectrical initComputer initHuaWei init

上面的Phone类和Computer类都继承了Electrical类,HuaWei类既继承了Phone类又继承了Computer类,这就形成了一个菱形的继承关系。

在创建HuaWei类对象时,Electrical类的__init__方法执行了两遍,也就是说在Phone类向上继承时执行了,在Computer类向上继承时也执行了,这显然是不应该发生的。

三、通过super解决初始化问题

class Electrical(object):    def __init__(self, name):        self.name = name        print('Electrical init')class Phone(Electrical):    def __init__(self, price, *args):        super(Phone, self).__init__(*args)        self.price = price        print('Phone init')class Computer(Electrical):    def __init__(self, config, *args):        super(Computer, self).__init__(*args)        self.config = config        print('Computer init')class HuaWei(Phone, Computer):    def __init__(self, name, price, config):        super(HuaWei, self).__init__(name, price, config)        print('HuaWei init')h = HuaWei('huawei', 100, 'i7')

运行结果:

Electrical initComputer initPhone initHuaWei init

通过super方法,我们解决了Electrical初始化两次的问题,现在的继承关系就是正常的了。

注意:在Phone类和Computer类中给super()后的init方法传参数时,应使用*args,因为HuaWei有三个参数,但是Phone类和Computer类都只有两个参数,所以参数个数不正确会报错。

四、菱形继承的继承顺序

class Electrical(object):    def chat(self):        print('Chat with friend in electrical!')    def watch_movie(self):        print('Watch movie in electrical!')    def game(self):        print('Play game in electrical!')class Phone(Electrical):    def game(self):        print('Play game in phone!')class Computer(Electrical):    def watch_movie(self):        print('Watch movie in computer!')    def game(self):        print('Play game in computer!')class HuaWei(Phone, Computer):    passh = HuaWei()h.game()h.watch_movie()h.chat()

运行结果:

Play game in phone!Watch movie in computer!Chat with friend in electrical!

在上面的继承关系中,HuaWei类中一个方法也没有实现,所以类对象调用方法都是要去父类中找。

game()方法在三个类中都实现了,调用了Phone类中的方法,说明最先会去Phone类中查找。

watch_movie()方法在Phone类中也没有实现,调用了Computer类中的方法,说明第二个会去Computer类中查找。

chat()方法在Phone类和Computer类中都没有实现,调用了Electrical类中的方法,说明最后会去Electrical类中查找。

注意:在继承时,如果小括号()中先写Computer再写Phone,则两个类的继承顺序就会调换。

由此,我们可以得出菱形继承的继承顺序了。如下图所示。

五、__mro__方法查看继承顺序

根据上面的案例,我们已经知道了菱形继承中的继承顺序了。

这种继承顺序是遵循广度优先算法的。也就是说,多层多继承时,先在父级中按先后顺序查找,然后在到父级的父级中查找。

这样,继承顺序已经很清晰了,不过,这还需要我们人工来识别继承顺序。如果继承关系非常复杂,像一张网一样,我们再去人工识别继承顺序将会非常困难,并且容易出错。

在Python中,已经定义了一个魔法方法来帮助我们查看类的继承顺序,这个方法就是__mro__方法。

print(HuaWei.__mro__)

运行结果:

(
,
,
,
,
)

mro是 Method Resolution Order (方法解析顺序)的简写,我们可以直接打印类的__mro__方法来获取类的继承顺序。

注意:__mro__只能通过类对象来使用,不能通过实例对象来使用。

 

 

转载地址:http://myppz.baihongyu.com/

你可能感兴趣的文章
NHibernate学习[1]
查看>>
NHibernate异常:No persister for的解决办法
查看>>
NIFI1.21.0_java.net.SocketException:_Too many open files 打开的文件太多_实际操作---大数据之Nifi工作笔记0051
查看>>
NIFI1.21.0_Mysql到Mysql增量CDC同步中_日期类型_以及null数据同步处理补充---大数据之Nifi工作笔记0057
查看>>
NIFI1.21.0_Mysql到Mysql增量CDC同步中_补充_更新时如果目标表中不存在记录就改为插入数据_Postgresql_Hbase也适用---大数据之Nifi工作笔记0059
查看>>
NIFI1.21.0_NIFI和hadoop蹦了_200G集群磁盘又满了_Jps看不到进程了_Unable to write in /tmp. Aborting----大数据之Nifi工作笔记0052
查看>>
NIFI1.21.0最新版本安装_连接phoenix_单机版_Https登录_什么都没改换了最新版本的NIFI可以连接了_气人_实现插入数据到Hbase_实际操作---大数据之Nifi工作笔记0050
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_增删改数据分发及删除数据实时同步_通过分页解决变更记录过大问题_02----大数据之Nifi工作笔记0054
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_插入修改删除增量数据实时同步_通过分页解决变更记录过大问题_01----大数据之Nifi工作笔记0053
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表或全表增量同步_实现指定整库同步_或指定数据表同步配置_04---大数据之Nifi工作笔记0056
查看>>
NIFI1.23.2_最新版_性能优化通用_技巧积累_使用NIFI表达式过滤表_随时更新---大数据之Nifi工作笔记0063
查看>>
NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_根据binlog实现数据实时delete同步_实际操作04---大数据之Nifi工作笔记0043
查看>>
NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置binlog_使用处理器抓取binlog数据_实际操作01---大数据之Nifi工作笔记0040
查看>>
NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置数据路由_实现数据插入数据到目标数据库_实际操作03---大数据之Nifi工作笔记0042
查看>>
NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_配置数据路由_生成插入Sql语句_实际操作02---大数据之Nifi工作笔记0041
查看>>
NIFI从MySql中离线读取数据再导入到MySql中_03_来吧用NIFI实现_数据分页获取功能---大数据之Nifi工作笔记0038
查看>>
NIFI从MySql中离线读取数据再导入到MySql中_不带分页处理_01_QueryDatabaseTable获取数据_原0036---大数据之Nifi工作笔记0064
查看>>
NIFI从MySql中离线读取数据再导入到MySql中_无分页功能_02_转换数据_分割数据_提取JSON数据_替换拼接SQL_添加分页---大数据之Nifi工作笔记0037
查看>>
NIFI从PostGresql中离线读取数据再导入到MySql中_带有数据分页获取功能_不带分页不能用_NIFI资料太少了---大数据之Nifi工作笔记0039
查看>>
nifi使用过程-常见问题-以及入门总结---大数据之Nifi工作笔记0012
查看>>