Python super()函数:调用父类的构造方法
创始人
2024-05-27 02:04:39
0

Python 中子类会继承父类所有的类属性和类方法。严格来说,类的构造方法其实就是实例方法,因此毫无疑问,父类的构造方法,子类同样会继承。

但我们知道,Python 是一门支持多继承的面向对象编程语言,如果子类继承的多个父类中包含同名的类实例方法,则子类对象在调用该方法时,会优先选择排在最前面的父类中的实例方法。显然,构造方法也是如此。

举个例子:

class People:def __init__(self,name):self.name = namedef say(self):print("我是人,名字为:",self.name)
class Animal:def __init__(self,food):self.food = fooddef display(self):print("我是动物,我吃",self.food)
#People中的 name 属性和 say() 会遮蔽 Animal 类中的
class Person(People, Animal):pass
per = Person("zhangsan")
per.say()
#per.display()

运行结果,结果为:

我是人,名字为: zhangsan

上面程序中,Person 类同时继承 People 和 Animal,其中 People 在前。这意味着,在创建 per 对象时,其将会调用从 People 继承来的构造函数。因此我们看到,上面程序在创建 per 对象的同时,还要给 name 属性进行赋值。

但如果去掉最后一行的注释,运行此行代码,Python 解释器会报如下错误:

Traceback (most recent call last):File "D:\python3.6\Demo.py", line 18, in per.display()File "D:\python3.6\Demo.py", line 11, in displayprint("我是动物,我吃",self.food)
AttributeError: 'Person' object has no attribute 'food'

这是因为,从 Animal 类中继承的 display() 方法中,需要用到 food 属性的值,但由于 People 类的构造方法“遮蔽”了Animal 类的构造方法,使得在创建 per 对象时,Animal 类的构造方法未得到执行,所以程序出错。

反过来也是如此,如果将第 13 行代码改为如下形式:

class Person(Animal, People)

则在创建 per 对象时,会给 food 属性传值。这意味着,per.display() 能顺序执行,但 per.say() 将会报错。

针对这种情况,正确的做法是定义 Person 类自己的构造方法(等同于重写第一个直接父类的构造方法)。但需要注意,如果在子类中定义构造方法,则必须在该方法中调用父类的构造方法。

在子类中的构造方法中,调用父类构造方法的方式有 2 种,分别是:

  1. 类可以看做一个独立空间,在类的外部调用其中的实例方法,可以向调用普通函数那样,只不过需要额外备注类名(此方式又称为未绑定方法);
  2. 使用 super() 函数。但如果涉及多继承,该函数只能调用第一个直接父类的构造方法。

也就是说,涉及到多继承时,在子类构造函数中,调用第一个父类构造方法的方式有以上 2 种,而调用其它父类构造方法的方式只能使用未绑定方法。

值得一提的是,Python 2.x 中,super() 函数的使用语法格式如下:

super(Class, obj).__init__(...)

其中,Class 值得是子类的类名,obj 通常指的就是 self。

但在 Python 3.x 中,super() 函数有一种更简单的语法格式,推荐大家使用这种格式:

super().__init__(...)

在掌握 super() 函数用法的基础上,我们可以尝试修改上面的程序:

class People:def __init__(self,name):self.name = namedef say(self):print("我是人,名字为:",self.name)
class Animal:def __init__(self,food):self.food = fooddef display(self):print("我是动物,我吃",self.food)
class Person(People, Animal):#自定义构造方法def __init__(self,name,food):#调用 People 类的构造方法super().__init__(name)#super(Person,self).__init__(name) #执行效果和上一行相同#People.__init__(self,name)#使用未绑定方法调用 People 类构造方法#调用其它父类的构造方法,需手动给 self 传值Animal.__init__(self,food)    
per = Person("zhangsan","熟食")
per.say()
per.display()

运行结果为:

我是人,名字为: zhangsan
我是动物,我吃 熟食

可以看到,Person 类自定义的构造方法中,调用 People 类构造方法,可以使用 super() 函数,也可以使用未绑定方法。但是调用 Animal 类的构造方法,只能使用未绑定方法。

Python教程,8天python从入门到精通,学python看这套就够了

相关内容

热门资讯

天津静海有个“萨克斯村”   本报记者 韩梅 通讯员 陈泽宇  华丽震撼的《巴比伦河》,撩人心弦的《回家》,如泣如诉的《友谊地...
铁路端午假期火车票昨日开售   本报讯(记者 胡子傲)记者从国铁集团获悉,铁路端午假期运输自5月30日至6月3日,为期5天,预计...
头顶千亿负债 东方盛虹202... 转自:中国经营报本报记者 李哲 北京报道近日,东方盛虹(000301.SZ)披露的2024年财报显示...
敦促美方停止对中国科技企业和人...   新华社北京5月16日电(记者董雪 刘杨)外交部发言人林剑16日表示,美方无端对中国芯片产品和人工...
美的旗下安得智联赴港上市 仍... 转自:中国经营报本报记者 陈靖斌 广州报道日前,美的集团拟对旗下安得智联供应链科技进行拆分,并赴港上...
终于来了!阿里最新财报发布,结... 运营商财经 康钊/文5月15日晚间,阿里发布2025财年Q4及全年财报,整体来看,最新年度的业绩数...
深入做好金融“五篇大文章” 为...   本报讯(记者 祁梦竹 刘菲菲)昨天上午,市委书记尹力围绕“服务国家金融管理中心建设,推动首都金融...
多个“首次”进一步激发市场活力   中国证监会16日公布实施修订后的《上市公司重大资产重组管理办法》,在简化审核程序、创新交易工具、...
山东矿机向实控人之子定增募资 ... 本报记者 庄灵辉 卢志坤 北京报道山东矿机(002526.SZ)向其实控人之子定增募资一事迎来新进展...
沙特阿美全球董事会会议在北京C...   本报讯(记者 朱松梅 通讯员 赵雷)日前,沙特阿拉伯国家石油公司(以下简称“沙特阿美”)全球董事...
广东多城公积金利率下调 大湾... 本报记者 陈婷 赵毅 深圳报道2025年5月8日起,广州、深圳、佛山、东莞、珠海、潮州、江门等广东省...
首次公布!我国空间站内发现微生... 近日,科研人员首次公布在我国空间站发现的一个微生物新物种,并将其命名为“天宫尼尔菌(Niallia ...
前4月京津冀区域出口创历史新高   本报讯(记者 鹿杨)记者日前从北京海关获悉,今年前4个月,京津冀区域进出口总额1.43万亿元人民...
现代化首都都市圈生机勃勃 九星科技为医院定制智慧监测大屏。受访者供图22号线建设现场的指挥中心。受访者供图  本报记者 李如意...
国际原油期货结算价涨超1% 本... 转自:财联社【国际原油期货结算价涨超1% 本周累涨超2%】财联社5月17日电,国际原油期货结算价涨超...
“降息潮”席卷民营银行 本报记者 慈玉鹏 北京报道步入5月,民营银行存款降息潮持续。《中国经营报》记者不完全统计,5月以来,...
短期高息理财产品“吸金” 银行... 近日,新发银行理财产品数量增加,部分理财机构密集推出高预期收益的最短持有期产品“吸金”。从产品设计来...
长安信托追讨11.98亿元投债... 近日,天风证券(601162.SH)一则公告披露了其与长安信托之间高达11.98亿元的证券交易合同纠...
明亚保险经纪控制权易主 团队... 本报记者 陈晶晶 北京报道近日,一项人事变动激起“千层浪”。根据国家金融监管总局北京监管局行政许可批...
破解确权与估值难题 首单数据... 本报记者 蒋牧云 李晖 上海 北京报道数据资产证券化的路径已经打开。近日,深圳证券交易所(以下简称“...