- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在研究菱形继承(钻石问题)并提出一个问题:
class A:
def __init__(self):
print("This is class A")
class B(A):
def __init__(self):
print("This is class B")
super().__init__()
class C(A):
def __init__(self):
print("This is class C")
super().__init__()
class D(B, C):
def __init__(self):
print("This is class D")
super().__init__()
i = D()
This is class D
This is class B
This is class C
This is class A
它按预期工作,这很好,但我想知道为什么 super().__init__()
在class B
不会转到 class A
而是调用 C。
如果一个类有 super() 并且它继承自父类,那么它应该去那里。
如果我在 B 上删除它,代码将不会到达 C 或 A。
我了解 MRO 以及它实际上如何按预期进行:
>>> D.__mro__
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)
但我不知道为什么。
这很奇怪,这段代码的非 super() 实现具有相同的 MRO,但 A 打印了两次:
class A:
def __init__(self):
print("This is class A")
class B(A):
def __init__(self):
print("This is class B")
A.__init__(self)
class C(A):
def __init__(self):
print("This is class C")
A.__init__(self)
class D(B, C):
def __init__(self):
print("This is class D")
B.__init__(self)
C.__init__(self)
i = D()
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)
情况恰恰相反,我知道 MRO 是正确的,但奇怪的是实际执行却并非如此:
This is class D
This is class B
This is class A
This is class C
This is class A
我想知道 super() 行为背后的逻辑是什么。
当在网上询问这个问题时,几乎每个人都将我链接到此:https://rhettinger.wordpress.com/2011/05/26/super-considered-super/但我真的不明白,他的解释似乎太技术性了,他的例子(我理解的几个例子)比解释这一点要复杂得多……这就是为什么我想要一个……更简单的解释。
即使父类的继承另有建议,Super() 也必须遵循 MRO?
Super() 无法转到父类的父类,因此如果父类中有 super,它将转到第二个继承类?
此外,有点不相关,但在真实的工作环境中看到菱形继承(钻石问题)有多常见?看起来是一种非常复杂的工作方式。
最佳答案
您需要记住,MRO 不仅仅是简单的追随领导者。它创建类似图形的结构并解析相同的继承以避免重复。在第一个示例中,您创建了菱形继承(钻石问题)。
A
/ \
B C
\ /
D
MRO 从第一级(直接父类)开始寻求方法解析,从左到右(B
然后 C
),然后是下一个级别,从左到右(这里只是 A
),依此类推。
在本例中,由于 B
和 C
均继承自 A
,因此顶层解析为单个 A
并创建上面的菱形图。
让我们看一下第二个示例:
class D(B, C):
def __init__(self):
print("This is class D")
B.__init__(self)
C.__init__(self)
通过这种方式实现,您可以有效地绕过 MRO。您已将继承钻石制成继承橄榄叉,如下所示:
A A
| |
B C
\ /
D
因此,您将调用 A
的初始化两次,这是不需要发生的。在长继承链或复杂的初始化例程中,这是非常低效的。
关于python - 为什么super()继承 "wrong"类?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59975088/
这个问题在这里已经有了答案: java.lang.IllegalArgumentException: The servlets named [X] and [Y] are both mapped t
我无法让我的 WebRTC 代码正常工作。我相信我所做的一切都是正确的,但它仍然无法正常工作。为什么 ontrack 这么早就被调用有些奇怪,也许它应该是那样的。 该网站使用 javascript 代
使用 Mac OSX 优胜美地 (10.10.4): rails -v => Rails 4.2.3 ruby -v => ruby 2.2.2p95 遵循这些说明的组合: https://www
您好,我正在尝试使用(缓存的)已编译的 lambda 表达式来访问属性,与使用 PropertyInfo.GetValue()/SetValue() 方法调用相比,我确实得到了更好(即更快)的结果。然
我编写此代码是为了获取学生的字母成绩并计算他们的 GPA。当我运行该程序时,我可以正确获取学生的姓名和科目,但无法显示成绩或 GPA。 示例输入: Sally 1 A N 示例输出: Enter St
我一直在编写这段代码,根据这本书应该可以做到这一点: Write a script that creates and calls a stored procedure named test. This
我真的很难创建一个具有以下基本格式的有效多维 JavaScript 数组: var countries = [ { "country": "UK", "properties": {
我有一个小型 Python OOP 程序,其中 2 个类 Flan 和 Outil 继承自父类(super class) Part。 我的问题是,当我调用 Flan 时一切正常,但是当我调用 Outi
我目前正在尝试使用通用监听器来编写事件系统。 所有监听器都应添加到单个 EventSource 对象,该对象将为特定事件调用其 receiveEvent() 方法。 事件源: public class
我正在通过我的 PHP 应用程序发送电子邮件。但是,它们被 Gmail 标记为垃圾邮件。这是我发送电子邮件的方式(PHP): $headers = "From: test@bookmytakeout.
我已经正式走到了穷途末路的地步。我找不到我做错了什么。我完成的这个程序几乎与我几天前编写的另一个程序一模一样,但我在编译时遇到了问题。我不知道为什么输出线上出现错误。请帮忙: 这是正在运行的文件: p
---编辑:我不允许使用任何包或预置方法。不用担心,我不想让你做我的“作业”,我只需要一点提示!---我发现these interesting Algorithms 。我想使用按位添加方法。我的问题是
我制作了一个小程序,尝试使用 conn.getOutputStream(); 检索 URLConnection 对象输出流。当我尝试执行此操作时,我的小程序抛出异常 java.net.UnknownS
每当我尝试在 SVN 中合并时,我都会遇到成堆的树冲突。好吧,就此示例脚本而言,只有一个,但仍然如此。 #!/bin/bash svnadmin create repo svn checkout fi
我开始为 Scala 中的 X500PrincipalBuilder 类编写单元测试。这是我的测试代码: import org.junit.runner.RunWith import org.scal
我正在用 python 编写我的第一个程序,它必须模拟粒子(两种气体)的混合。我不知道我这个功能做错了什么。我不希望颗粒离开某些区域,即容器的壁。我使用 VPython。 def poruszanie
我正在尝试求解三角方程组,但我认为 Python 没有生成正确的解。我试图解决的方程: 1 − 2cosθ1 + 2cosθ2 − 2cosθ3 = −0.8 1 − 2cos5θ1 + 2cos5θ
这个问题已经有答案了: TypeError: worker() takes 0 positional arguments but 1 was given [duplicate] (11 个回答) 已关
大家好,我正在努力解决这个问题 编写一个 C 程序,计算弹丸在撞击地面之前行进的距离(即射程)、弹丸撞击地面所需的时间以及弹丸飞行中的最大高度(给定角度)它被射向空中,以及发射时的初始速度(速度)。我
我编写了代码来计算 QuickSort 中完成的比较次数。 每当对长度为 m 的数组执行快速排序时,该算法都会将比较次数增加 m-1(因为主元将与除自身以外的所有内容进行比较)。 枢轴的选择始终是数组
我是一名优秀的程序员,十分优秀!