class A(object): # A must be new-style class def __init__(self): print "enter A" print "leave A" class B(C): # A --> C def __init__(self): print "enter B" super(B, self).__init__() print "leave B"
우리 생각에 super(B, self).__init__()는 다음과 같이 이해됩니다. super(B, self)는 먼저 B의 상위 클래스(즉, 클래스 A)를 찾은 다음 클래스를 넣습니다. B into 객체 self는 클래스 A의 객체로 변환되고, 그런 다음 "변환된" 클래스 A 객체는 자신의 __init__ 함수를 호출합니다.
어느 날 동료가 비교적 복잡한 수업 시스템 구조를 설계했는데(수업 시스템 설계가 합리적인지 아닌지 고민하지 말고 이 예제를 주제로 공부하세요) 코드는 다음과 같습니다
코드 세그먼트 4:
class A(object): def __init__(self): print "enter A" print "leave A" class B(object): def __init__(self): print "enter B" print "leave B" class C(A): def __init__(self): print "enter C" super(C, self).__init__() print "leave C" class D(A): def __init__(self): print "enter D" super(D, self).__init__() print "leave D" class E(B, C): def __init__(self): print "enter E" B.__init__(self) C.__init__(self) print "leave E" class F(E, D): def __init__(self): print "enter F" E.__init__(self) D.__init__(self) print "leave F"
f = F(), 결과는 다음과 같습니다.
enter F Enter E Enter B Leave B Enter C Enter D Enter A Leave A Leave D Leave C Leave E Enter D Enter A Leave A Leave D Leave F
분명히 클래스 A와 클래스 D의 초기화 함수는 두 번 호출되는 것이 아닙니다. 원하는 결과! 예상되는 결과는 클래스 A의 초기화 함수가 최대 두 번 호출된다는 것입니다. 실제로 이는 다중 상속 클래스 시스템이 직면해야 하는 문제입니다. 우리는 아래와 같이 코드 세그먼트 4의 클래스 시스템을 그립니다:
object
| E |
|
F
super에 대한 우리의 이해에 따르면, 그림을 보면 C 클래스의 초기화 함수를 호출할 때 A 클래스의 초기화 함수를 호출해야 하는데 실제로는 D 클래스의 초기화 함수를 호출하고 있는 것을 알 수 있다. 정말 이상한 질문입니다!
즉, mro는 클래스의 모든 기본 클래스의 클래스 유형 순서를 기록합니다. mro의 기록을 보면 7개의 요소가 포함되어 있고 7개의 클래스 이름은 다음과 같습니다.
Enter F Enter E Enter B Enter C Enter D Enter A Leave A Leave D Leave C Leave B Leave E Leave F
class A(object): def __init__(self): print "enter A" super(A, self).__init__() # new print "leave A" class B(object): def __init__(self): print "enter B" super(B, self).__init__() # new print "leave B" class C(A): def __init__(self): print "enter C" super(C, self).__init__() print "leave C" class D(A): def __init__(self): print "enter D" super(D, self).__init__() print "leave D" class E(B, C): def __init__(self): print "enter E" super(E, self).__init__() # change print "leave E" class F(E, D): def __init__(self): print "enter F" super(F, self).__init__() # change print "leave F"
요약
1. super는 함수가 아니라 클래스 이름입니다. super(B, self) 형식은 실제로 슈퍼 클래스의 초기화 함수를 호출합니다.
슈퍼 객체가 생성됩니다.
3. super(B, self) 호출. func는 현재 클래스의 상위 클래스의 func 함수를 호출하는 데 사용되지 않습니다.
4. Python의 다중 상속 클래스는 mro를 사용하여 각 상위 클래스의 함수가 하나씩 호출되도록 합니다. 각 상위 클래스 함수 한 번만 호출됩니다(모든 클래스가 super를 사용하는 경우)
5. 슈퍼 클래스와 바인딩 해제된 함수를 혼합하는 것은 위험한 동작으로, 호출되어야 하는 상위 클래스 함수가 호출되지 않거나 호출되지 않을 수 있습니다.
상위 클래스 함수가 여러 번 호출됩니다.
좀 더 심층적인 질문: 보시다시피 F.__mro__를 인쇄할 때 내부 요소의 순서가 F E B C D A 개체임을 알 수 있습니다. 이는 F의 기본 클래스 검색 순서입니다. 왜 이 순서인지, python 내장 다중 상속 시퀀스를 구현하는 방법에는 mro 시퀀스 구현이 포함됩니다. Python 2.3 이후 버전에서는 C3이라는 알고리즘을 사용하는데, 이에 대해서는 다음 블로그에서 소개하겠습니다.