@Visionscaperの上部に追加します。
Third --> First --> object --> Second --> object
この場合、インタプリタはオブジェクトクラスを除外しません。オブジェクトクラスが重複しているからではなく、Secondが階層サブセットの先頭位置に表示され、末尾位置に表示されないためです。オブジェクトはテール位置にのみ表示され、優先度を決定するC3アルゴリズムでは強い位置とは見なされません。
クラスCの線形化(mro)、L(C)は、
- クラスC
- プラスのマージ
- 親の線形化P1、P2、.. = L(P1、P2、...)および
- 親のリストP1、P2、..
線形化されたマージは、順序が重要であるため、末尾ではなくリストの先頭として表示される共通クラスを選択することによって行われます(以下で明らかになります)。
Thirdの線形化は次のように計算できます。
L(O) := [O] // the linearization(mro) of O(object), because O has no parents
L(First) := [First] + merge(L(O), [O])
= [First] + merge([O], [O])
= [First, O]
// Similarly,
L(Second) := [Second, O]
L(Third) := [Third] + merge(L(First), L(Second), [First, Second])
= [Third] + merge([First, O], [Second, O], [First, Second])
// class First is a good candidate for the first merge step, because it only appears as the head of the first and last lists
// class O is not a good candidate for the next merge step, because it also appears in the tails of list 1 and 2,
= [Third, First] + merge([O], [Second, O], [Second])
// class Second is a good candidate for the second merge step, because it appears as the head of the list 2 and 3
= [Third, First, Second] + merge([O], [O])
= [Third, First, Second, O]
したがって、次のコードのsuper()実装の場合:
class First(object):
def __init__(self):
super(First, self).__init__()
print "first"
class Second(object):
def __init__(self):
super(Second, self).__init__()
print "second"
class Third(First, Second):
def __init__(self):
super(Third, self).__init__()
print "that's it"
この方法がどのように解決されるかが明らかになる
Third.__init__() ---> First.__init__() ---> Second.__init__() --->
Object.__init__() ---> returns ---> Second.__init__() -
prints "second" - returns ---> First.__init__() -
prints "first" - returns ---> Third.__init__() - prints "that's it"
super()
です。無駄なオーバーヘッドである線形継承を使用するクラスで使用することはお勧めしません。