python多继承构造函数声明问题

news2024/11/28 15:59:32

前言:

嗨喽~大家好呀,这里是魔王呐 ❤ ~!

python更多源码/资料/解答/教程等 点击此处跳转文末名片免费获取

背景

有场景分别定义两组逻辑,随后有统一入口做基类属性的整合

其中两组逻辑的积累构造函数定义入参不同

设计类继承图如:

  • 实际的使用方式抽象为[使用] 小节

  • 实际开发过程中遇到问题

先说结论

  1. python 多继承,需要使用super函数进行MRO的依次不重复初始化

  2. python 多继承的情况下,构造函数__init__会被依次调用并传递参数

  3. python 多继承情况下,__init__参数需要保持一致,否则会出现某些继承路径上的基类初始化遇到异常

  4. python 多继承情况下,若构造函数参数不一致,可通过(*args, **kwargs)来统一

  5. python 多继承情况下,若有公共基类,MRO可被调整为有跳跃路径,进而利用子类不同的构造函数完成正常初始化,但需要临近基类可以处理子类传来的所有参数。

使用

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
# tjc = TestJobConfiger()
tjc = SubTest()
print vars(tjc)
print "\n".join([tjc.base_key,
                tjc.base1_key,
                tjc.subbase_key,
                tjc.subbase1_key,
                tjc.subtest_key])

期望结果

enter Base
enter SubBase
enter Base1|_arg1 |_arg2 
enter SubBase1|_arg1 |_arg2 
vars : {'base1_key': 'base1_key', 'subtest_key': 'subtest_key', 'subbase1_key': 'subbase1_key', 'subbase_key': 'subbase_key', 'base_key': 'base_key'}
values: base_key
base1_key
subbase_key
subbase1_key
subtest_key

第一版实现

class Base(object):
    def __init__(self):
        print "enter Base"
        self.base_key = "base_key"

class Base1(object):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        print "enter Base1" + "|" +  _arg1 + "|" + _arg2
        self.base1_key = "base1_key"

class SubBase(Base):
    def __init__(self):
        super(SubBase, self).__init__()
        print "enter SubBase"
        self.subbase_key = "subbase_key"

class SubBase1(Base1):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(SubBase1, self).__init__(_arg1 = _arg1, _arg2=_arg2)
        print "enter SubBase1" + "|" +  _arg1 + "|" + _arg2
        self.subbase1_key = "subbase1_key"

class SubTest(SubBase,SubBase1):
    def __init__(self, _arg1 = "_arg1 "):
        super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
        # self.__dict__.update(vars(SubBase()))
        self.subtest_key = "subtest_key"

运行结果为:

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
Traceback (most recent call last):
  File "/Users/enzhao/suanec/ksp/dispatch/weiclient/client/weiclient/libs/com/weibo/tools/job_manager/job_configer_tester.py", line 43, in <module>
    tjc = SubTest()
  File "/Users/enzhao/suanec/ksp/dispatch/weiclient/client/weiclient/libs/com/weibo/tools/job_manager/job_configer_tester.py", line 38, in __init__
    super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
TypeError: __init__() got an unexpected keyword argument '_arg1'

第二版实现

class Base(object):
    def __init__(self):
        print "enter Base"
        self.base_key = "base_key"

class Base1(object):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        print "enter Base1" + "|" +  _arg1 + "|" + _arg2
        self.base1_key = "base1_key"

class SubBase(Base):
    def __init__(self, **kwargs):
        super(SubBase, self).__init__()
        print "enter SubBase"
        self.subbase_key = "subbase_key"

class SubBase1(Base1):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(SubBase1, self).__init__(_arg1 = _arg1, _arg2=_arg2)
        print "enter SubBase1" + "|" +  _arg1 + "|" + _arg2
        self.subbase1_key = "subbase1_key"

class SubTest(SubBase,SubBase1):
    def __init__(self, _arg1 = "_arg1 "):
        super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
        # self.__dict__.update(vars(SubBase()))
        self.subtest_key = "subtest_key"

运行结果为:

Traceback (most recent call last):
enter Base
enter SubBase
  File "/Users/enzhao/suanec/ksp/dispatch/weiclient/client/weiclient/libs/com/weibo/tools/job_manager/job_configer_tester.py", line 46, in <module>
vars : {'subtest_key': 'subtest_key', 'subbase_key': 'subbase_key', 'base_key': 'base_key'}
    tjc.base1_key,
AttributeError: 'SubTest' object has no attribute 'base1_key'

二次继承实现

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class Base(object):
    def __init__(self):
        print "enter Base"
        self.base_key = "base_key"

class Base1(Base):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(Base1, self).__init__()
        print "enter Base1" + "|" +  _arg1 + "|" + _arg2
        self.base1_key = "base1_key"

class SubBase(Base):
    def __init__(self, **kwargs):
        super(SubBase, self).__init__()
        print "enter SubBase"
        self.subbase_key = "subbase_key"

class SubBase1(Base1):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(SubBase1, self).__init__(_arg1 = _arg1, _arg2=_arg2)
        print "enter SubBase1" + "|" +  _arg1 + "|" + _arg2
        self.subbase1_key = "subbase1_key"

class SubTest(SubBase,SubBase1):
    def __init__(self, _arg1 = "_arg1 "):
        super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
        # self.__dict__.update(vars(SubBase()))
        self.subtest_key = "subtest_key"

运行结果为:

enter Base
enter Base1|_arg1 |_arg2 
enter SubBase1|_arg1 |_arg2 
enter SubBase
vars : {'base1_key': 'base1_key', 'subtest_key': 'subtest_key', 'subbase_key': 'subbase_key', 'subbase1_key': 'subbase1_key', 'base_key': 'base_key'}
values: base_key
base1_key
subbase_key
subbase1_key
subtest_key

公共基类实现

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class BBase(object):
    def __init__(self):
        pass

class Base(BBase):
    def __init__(self):
        print "enter Base"
        self.base_key = "base_key"

class Base1(BBase):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        print "enter Base1" + "|" +  _arg1 + "|" + _arg2
        self.base1_key = "base1_key"

class SubBase(Base):
    def __init__(self, **kwargs):
        super(SubBase, self).__init__()
        print "enter SubBase"
        self.subbase_key = "subbase_key"

class SubBase1(Base1):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(SubBase1, self).__init__(_arg1 = _arg1, _arg2=_arg2)
        print "enter SubBase1" + "|" +  _arg1 + "|" + _arg2
        self.subbase1_key = "subbase1_key"

class SubTest(SubBase,SubBase1):
    def __init__(self, _arg1 = "_arg1 "):
        super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
        # self.__dict__.update(vars(SubBase()))
        self.subtest_key = "subtest_key"

运行结果为:

enter Base
enter SubBase
vars : {'subtest_key': 'subtest_key', 'subbase_key': 'subbase_key', 'base_key': 'base_key'}
Traceback (most recent call last):
  File "/Users/enzhao/suanec/ksp/dispatch/weiclient/client/weiclient/libs/com/weibo/tools/job_manager/job_configer_tester.py", line 50, in <module>
    tjc.base1_key,
AttributeError: 'SubTest' object has no attribute 'base1_key'

Hack实现

分别保持各自定义逻辑

在入口子类中,声明冲突基类的对象,利用python的vars和__dict__的特性进行属性的声明

相当于手动hard-code 构造函数的调用,完成基类的初始化

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class Base(object):
    def __init__(self):
        print "enter Base"
        self.base_key = "base_key"

class Base1(object):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        print "enter Base1" + "|" +  _arg1 + "|" + _arg2
        self.base1_key = "base1_key"

class SubBase(Base):
    def __init__(self):
        super(SubBase, self).__init__()
        print "enter SubBase"
        self.subbase_key = "subbase_key"

class SubBase1(Base1):
    def __init__(self, _arg1 = "_arg1 ", _arg2 = "_arg2 "):
        super(SubBase1, self).__init__(_arg1 = _arg1, _arg2=_arg2)
        print "enter SubBase1" + "|" +  _arg1 + "|" + _arg2
        self.subbase1_key = "subbase1_key"

class SubTest(SubBase1):
    def __init__(self, _arg1 = "_arg1 "):
        super(SubTest, self).__init__(_arg1=_arg1, _arg2="None")
        self.__dict__.update(vars(SubBase()))
        self.subtest_key = "subtest_key"

运行结果为:

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
enter Base1|_arg1 |None
enter SubBase1|_arg1 |None
enter Base
enter SubBase
vars : {'base1_key': 'base1_key', 'subtest_key': 'subtest_key', 'subbase_key': 'subbase_key', 'subbase1_key': 'subbase1_key', 'base_key': 'base_key'}
values: base_key
base1_key
subbase_key
subbase1_key
subtest_key

官方建议实现

class A(object):
    def __init__(self, *args, **kwargs):
        print "A"

class B(object):
    def __init__(self, *args, **kwargs):
        print "B"

class C(A):
    def __init__(self, arg, *args, **kwargs):
        print "C","arg=",arg
        super(C, self).__init__(arg, *args, **kwargs)

class D(B):
    def __init__(self, arg, *args, **kwargs):
        print "D", "arg=",arg
        super(D, self).__init__(arg, *args, **kwargs)

class E(C,D):
    def __init__(self, arg, *args, **kwargs):
        print "E", "arg=",arg
        super(E, self).__init__(arg, *args, **kwargs)

print "MRO:", [x.__name__ for x in E.__mro__]
E(10)
MRO: ['E', 'C', 'A', 'D', 'B', 'object']
E arg= 10
C arg= 10
A

尾语

最后感谢你观看我的文章呐~本次航班到这里就结束啦 🛬

希望本篇文章有对你带来帮助 🎉,有学习到一点知识~

躲起来的星星🍥也在努力发光,你也要努力加油(让我们一起努力叭)。

最后,宣传一下呀~👇👇👇更多源码、资料、素材、解答、交流皆点击下方名片获取呀👇👇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1041933.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【深度学习】【Opencv】Python/C++调用onnx模型【基础】

【深度学习】【Opencv】python/C调用onnx模型【基础】 提示:博主取舍了很多大佬的博文并亲测有效,分享笔记邀大家共同学习讨论 文章目录 【深度学习】【Opencv】python/C调用onnx模型【基础】前言Python版本OpenCVWindows平台安装OpenCVopencv调用onnx模型 C版本OpenCVWindows平…

终极策略:如何利用亮数据代理轻松、高效地突破亚马逊的反爬障碍

文章目录 前言背景&#x1f4dc;第一步&#xff1a;打开亚马逊商城&#x1f6cd;️第二步&#xff1a;定位搜索框并搜索iphone15&#x1f50d;第三步&#xff1a;定位具体数据并保存到csv文件&#x1f4be;第三步&#xff1a;多页面数据抓取&#x1f4c4;&#x1f4c4;&#x1…

OpenCV 基础图像处理

1、生成图像 cv2.imread是OpenCV库中的一个函数&#xff0c;用于读取图像文件。它接受一个参数&#xff0c;即要读取的图像文件的路径&#xff0c;返回一个多维数组&#xff0c; 表示图像的像素值。该函数的常用参数包括&#xff1a;flags&#xff1a;指定读取图像的方式&#…

RabbitMQ学习总结(11)—— RabbitMQ 核心概念与架构

MQ 技术概述 什么是 MQ MQ(message queue),从字面意思上看,本质是个队列,FIFO 先入先出,只不过队列中存放的内容是 message 而已,还是一种跨进程的通信机制,用于上下游传递消息。在互联网架构中,MQ 是一种非常常见的上下游 “逻辑解耦+物理解耦” 的消息通信服务。使用…

系统集成|第十三章(笔记)

目录 第十三章 干系人管理13.1 概述与相关概念13.2 主要过程13.2.1 识别干系人13.2.2 编制项目干系人管理计划13.2.3 管理干系人参与13.2.4 项目干系人参与的监控 13.3 常见问题 上篇&#xff1a;第十二章、沟通管理 第十三章 干系人管理 13.1 概述与相关概念 概述&#xff1a…

21-SpringSecurity

SpringSecurity从入门到精通 0. 简介 ​ Spring Security 是 Spring 家族中的一个安全管理框架。相比与另外一个安全框架Shiro&#xff0c;它提供了更丰富的功能&#xff0c;社区资源也比Shiro丰富。 ​ 一般来说中大型的项目都是使用SpringSecurity 来做安全框架。小项目有…

会议剪影 | 思腾合力受邀参加2023第二届世界元宇宙大会并作主题演讲

由中国仿真学会、中国指挥与控制学会和北京理工大学共同主办&#xff0c;上海市嘉定区安亭镇人民政府和中国仿真学会元宇宙专业委员会承办的第二届世界元宇宙大会于2023年9月20日-22日在上海安亭举行。 大会以“虚实相生、产业赋能”为主题&#xff0c;聚焦元宇宙关键技术发展的…

Redis原理(一):Redis数据结构(上)

文章目录 1、 Redis数据结构-动态字符串2、 Redis数据结构-intset3、 Redis数据结构-Dict4、 Redis数据结构-ZipList5、 Redis数据结构-ZipList的连锁更新问题6、 Redis数据结构-QuickList1、 Redis数据结构-动态字符串 我们都知道Redis中保存的Key是字符串,value往往是字符串…

Mybatis 日志(Log4j2)

之前我们介绍了使用JDK Log、Apache Commons Logging打印Mybatis运行时的日志&#xff1b;本篇我们介绍使用Log4j2打印Mybatis运行时的日志。 如何您对Mybatis中使用JDK Log、Apache Commons Logging打印Mybatis运行时的日志不太了解&#xff0c;可以参考&#xff1a; Mybati…

使用群晖实现Videostation电影的大容量存储及分享教程

文章目录 1.使用环境要求2.制作视频分享链接3.制作永久固定视频分享链接 李哥和他的女朋友是一对甜蜜的情侣&#xff0c;但不幸的是&#xff0c;由于工作原因&#xff0c;他们目前分隔两地&#xff0c;无法常常亲密相伴。 这个距离让李哥特别怀念和女朋友一起在电影院观看电影的…

zabbix6.0监控磁盘IO

客户端配置 配置参数 [rootora19c zabbix]# pwd /usr/local/zabbix/etc/ [rootora19c zabbix]# vim zabbix_agentd.conf UnsafeUserParameters1配置UI监控conf文件 [rootora19c zabbix_agentd.d]# pwd /usr/local/zabbix/etc/zabbix_agentd.d/ [rootora19c zabbix_agentd.d…

大数据Flink(八十六):DML:Group 聚合和Over 聚合

文章目录 DML:Group 聚合和Over 聚合 一、DML:Group 聚合

CasaOS:一个docker容器应用的可视化Portal

CasaOS 官网声称他是一个家庭云操作系统&#xff0c;但我实际使用后感觉称之为“docker容器的可视化Portal”更合适。因为它本身不具备IAAS、PAAS、或SAAS的开箱即用能力&#xff0c;更像是一个把OS上的docker Container集中管理并展示的索引目录&#xff0c;各个docker Contai…

点成分享丨水浴设备在食品理化特性研究中的应用

木薯粉拯救“一麦难求” 近年来&#xff0c;随着全球人口增长、城市化进程的加快及人们饮食习惯的改变&#xff0c;以小麦为原料的食品成为了人们的“热门主食”&#xff0c;但实际每年产出的小麦不足以满足人们与日俱增的需求。为了解决这一问题&#xff0c;人们开始寻找小麦…

CTF_BUUCTF_Reverse解题_04内涵的软件

题目地址&#xff1a;BUUCTF在线评测 拿到文件&#xff0c;运行&#xff0c;狗&#xff01;&#xff01;&#xff01; 不管回答Y还是N&#xff0c;程序都最终会一闪而过 上PE 32位C没壳 上IDA 32 F5反汇编一下&#xff1a; 直接发现Y和N&#xff0c; 看到有if判断语…

安科瑞医用隔离电源系统在江苏某医院项目中的应用

安科瑞 崔丽洁 江苏摘要&#xff1a;介绍该三级乙等综合医院采用安科瑞医用隔离电源系统&#xff0c;使用分体配电柜安装方式&#xff0c;从而实现将TN系统转化为IT系统&#xff0c;以及系统绝缘情况监测。 1、概述 该医院占地面积4.2万平方米&#xff0c;建筑面积7.14万平方米…

【C语言】通讯录

目录 一、关于通讯录 二、代码逻辑 三、通讯录实现 1.菜单设计 2.逻辑主要功能设计 3.增加联系人功能实现 4.显示全部联系人信息 5.删除联系人 6.查找联系人 7.修改联系人信息 8.对联系人进行排序 9.一键清空所有联系人 四、完整源码 test.c contact.c contact.…

新版WordPress系统文章自动采集插件/Auto Post pro完美运行版/多线程采集(wp自动采集)

源码介绍&#xff1a; 最新版WordPress系统文章自动采集插件&#xff0c;它是一款帮助用户提供方便快捷的文章自动采集方案的插件。WordPress自动采集插件&#xff0c;让内容采集变得高效便捷。作为Auto Post pro完美运行版&#xff0c;这里分享的是WordPress文章采集插件Auto…

【Vue3】v-model

v-model 基本用法 prop: modelValue 事件&#xff1a;update:modelValue <!-- App.vue --><template><div><h1>我是父组件</h1><div>isShow: {{ isShow }}</div><div><button click"isShow !isShow">开关&…

Java 华为真题-出租车计费

需求 程序员小明打了一辆出租车去上班。出于职业敏感&#xff0c;他注意到这辆出租车的计费表有点问题&#xff0c;总是偏大。 出租车司机解释说他不喜欢数字4&#xff0c;所以改装了计费表&#xff0c;任何数字位置遇到数字4就直接跳过&#xff0c;其余功能都正常。 比如&…