Home > Backend Development > Python Tutorial > Python: super you don't know

Python: super you don't know

高洛峰
Release: 2016-11-15 15:14:37
Original
1198 people have browsed it

Getting started with super()

In class inheritance, if a method is redefined, the method will override the method of the same name of the parent class, but sometimes, we hope to realize the functions of the parent class at the same time. In this case, we It is necessary to call the method of the parent class, which can be achieved by using super, such as:

class Animal(object):
    def __init__(self, name):
        self.name = name
    def greet(self):
        print 'Hello, I am %s.' % self.name

class Dog(Animal):
    def greet(self):
        super(Dog, self).greet()   # Python3 可使用 super().greet()
        print 'WangWang...'
Copy after login

In the above, Animal is the parent class and Dog is the subclass. We redefined the greet method in the Dog class in order to realize the parent class at the same time. function, we call the method of the parent class again, see the following usage:

>>> dog = Dog('dog')
>>> dog.greet()
Hello, I am dog.
WangWang..
Copy after login

One of the most common uses of super can be said to be to call the initialization method of the parent class in the subclass, such as:

class Base(object):
    def __init__(self, a, b):
        self.a = a
        self.b = b

class A(Base):
    def __init__(self, a, b, c):
        super(A, self).__init__(a, b)  # Python3 可使用 super().__init__(a, b)
        self.c = c
Copy after login

Go deeper into super()

After reading the above usage, you may think that the use of super is very simple. It is nothing more than getting the parent class and calling the parent class’s method. In fact, in the above case, the class obtained by super happens to be the parent class, but in other cases, this is not necessarily the case. Super actually has no substantial relationship with the parent class.

Let us look at a slightly more complicated example involving multiple inheritance. The code is as follows:

class Base(object):
    def __init__(self):
        print "enter Base"
        print "leave Base"

class A(Base):
    def __init__(self):
        print "enter A"
        super(A, self).__init__()
        print "leave A"

class B(Base):
    def __init__(self):
        print "enter B"
        super(B, self).__init__()
        print "leave B"

class C(A, B):
    def __init__(self):
        print "enter C"
        super(C, self).__init__()
        print "leave C"
Copy after login

Among them, Base is the parent class, A and B inherit from Base, and C inherits from A and B. Their inheritance relationship is as follows:

      Base
      /  \
     /    \
    A      B
     \    /
      \  /
       C
Copy after login

Now, let’s take a look at the usage:

>>> c = C()
enter C
enter A
enter B
enter Base
leave Base
leave B
leave A
leave C
Copy after login

If you think super means “calling the method of the parent class”, then you are probably wondering why the next sentence of enter A is not enter Base but enter B. The reason is that super has no substantial relationship with the parent class. Now let us figure out how super works.

MRO list

In fact, for each class you define, Python will calculate a method resolution order (MRO) list, which represents the order of class inheritance. We can use the following method to obtain a method Class MRO list:

>>> C.mro()   # or C.__mro__ or C().__class__.mro()
[__main__.C, __main__.A, __main__.B, __main__.Base, object]
Copy after login

So how is the order of this MRO list determined? It is implemented through a C3 linearization algorithm. We will not delve into this algorithm here. Interested readers can learn about it themselves. To summarize, in general, the MRO list of a class is to merge the MRO lists of all parent classes, and follow the following three principles:

Subclasses are always in front of the parent class

If there are multiple parent classes, they will be placed in the list according to them The order in is checked

If there are two legal choices for the next class, the first parent class is selected

super principle

super works as follows:

def super(cls, inst):
    mro = inst.__class__.mro()
    return mro[mro.index(cls) + 1]
Copy after login

where, cls represents the class, inst represents the instance, The above code does two things:

Get the MRO list of inst

Find the index of cls in the current MRO list, and return its next class, which is mro[index + 1]

When you use super( cls, inst), Python will search for the next class of cls on the MRO list of inst.

Now, let’s go back to the previous example.

First look at the __init__ method of class C:

super(C, self).__init__()
Copy after login

The self here is the current instance of C, self.__class__.mro() The result is:

[__main__.C, __main__.A, __main__.B, __main__.Base, object]
Copy after login

You can see that the next class of C is A, so jump When A's __init__ is reached, enter A will be printed and the following line of code will be executed:

super(A, self).__init__()
Copy after login

Note that self here is also an instance of the current C. The MRO list is the same as above. Search for the next class of A in the MRO. , and found that it was B, so it jumped to B's __init__. At this time, enter B will be printed instead of enter Base.

The whole process is relatively clear. The key is to understand how super works, rather than taking it for granted that super calls the method of the parent class.

Summary

In fact, super has no substantial relationship with the parent class.

super(cls, inst) gets the next class of cls in the MRO list of inst.


Related labels:
source:php.cn
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template