Inheritance and super() - Object Oriented Programming
Table of Contents
_s_uper() is a feature through which inherited methods can be accessed, which has been overridden in a class. It can also help with the MRO lookup order in case of multiple inheritance. This may not be obvious first, but a few examples should help to drive the point home.
Inheritance and method overloading was discussed in a previous post, where we saw how inherited methods can be overloaded or enhanced in the child classes.
In many scenarios, it's needed to overload an inherited method, but also call the actual method defined in the Parent class.
Let's start off with a simple example based on Inheritance, and build from there.
Example 0:
[code language="python"] class MyClass(object):
def func(self): print("I'm being called from the Parent class!")
class ChildClass(MyClass): pass
my_instance_1 = ChildClass() my_instance_1.func() [/code] This outputs:
[code language="python"] In [18]: %run /tmp/super-1.py I'm being called from the Parent class [/code] In Example 0, we have two classes, MyClass
and ChildClass
. The latter inherits from the former, and the parent class MyClass
has a method named func
defined.
Since ChildClass
inherits from MyClass
, the child class has access to the methods defined in the parent class. An instance is created my_instance_2
, for ChildClass.
Calling my_instance_1.func()
will print the statement from the Parent class, due to the inheritance.
Building up on the first example:
Example 1:
[code language="python"] class MyClass(object):
def func(self): print("I'm being called from the Parent class")
class ChildClass(MyClass):
def func(self): print("I'm being called from the Child class")
my_instance_1 = MyClass() my_instance_2 = ChildClass()
my_instance_1.func() my_instance_2.func() [/code] This outputs:
[code language="python"] In [19]: %run /tmp/super-1.py I'm being called from the Parent class I'm being called from the Child class [/code] This example has a slight difference, both the child class as well as the parent class have the same method defined, ie.. func
. In this scenario, the parent class' method is overridden by the child class method.
ie.. if we call the func()
method from the instance of ChildClass
, it need not go a fetch the method from its Parent class, since it's already defined locally.
NOTE: This is due to the Method Resolution Order, discussed in an earlier post.
But what if there is a scenario that warranties the need for specifically calling methods defined in the Parent class, from the instance of a child class?
ie.. How to call the methods defined in the Parent class, through the instance of the Child class, even if the Parent class method is overloaded in the Child class?
In such a case, the inbuilt function super()
can be used. Let's add to the previous example.
Example 2:
[code language="python"] class MyClass(object):
def func(self): print("I'm being called from the Parent class")
class ChildClass(MyClass):
def func(self): print("I'm actually being called from the Child class") print("But...") # Calling the `func()` method from the Parent class. super(ChildClass, self).func()
my_instance_2 = ChildClass() my_instance_2.func() [/code] This outputs:
[code language="python"] In [21]: %run /tmp/super-1.py I'm actually being called from the Child class But... I'm being called from the Parent class [/code]
How is the code structured?#
- We have two classes
MyClass
andChildClass
. - The latter is inheriting from the former.
- Both classes have a method named
func
- The child class
ChildClass
is instantiated asmy_instance_2
- The
func
method is called from the instance.
How does the code work?#
- When the
func
method is called, the interpreter searches it using the Method Resolution Order, and find the method defined in the classChildClass
. - Since it finds the method in the child class, it executes it, and prints the string "I'm actually being called from the Child class", as well "But..."
- The next statement is
super
which calls the methodfunc
defined in the parent class ofChildClass
- Since the control is now passed onto the
func
method in the Parent class viasuper
, the correspondingprint()
statement is printed to stdout.
Example 2 can also be re-written as :
[code language="python"] class MyClass(object):
def func(self): print("I'm being called from the Parent class")
class ChildClass(MyClass):
def func(self): print("I'm actually being called from the Child class") print("But...") # Calling the `func()` method from the Parent class. # super(ChildClass, self).func() MyClass.func(self) # Call the method directly via Parent class
my_instance_2 = ChildClass() my_instance_2.func() [/code]
NOTE: The example above uses the Parent class directly to access it's method. Even though it works, it is not the best way to do it since the code is tied to the Parent class name. If the Parent class name changes, the child/sub class code has to be changed as well.
Let's see another example for super()
. This is from our previous article on
Inheritance and method overloading.
Example 3:
[code language="python"] import abc
class MyClass(object):
__metaclass__ = abc.ABCMeta
def my_set_val(self, value): self.value = value
def my_get_val(self): return self.value
@abc.abstractmethod def print_doc(self): return
class MyChildClass(MyClass):
def my_set_val(self, value): if not isinstance(value, int): value = 0 super(MyChildClass, self).my_set_val(self)
def print_doc(self): print("Documentation for MyChild Class")
my_instance = MyChildClass() my_instance.my_set_val(100) print(my_instance.my_get_val()) print(my_instance.print_doc()) [/code] The code is already discussed
here. The my_set_val
method is defined in both the child class as well as the parent class.
We overload the my_set_val
method defined in the parent class, in the child class. But after enhancing/overloading it, we call the my_set_val
method specifically from the Parent class using super()
and thus enhance it.
Takeaway:#
- super() helps to specifically call the Parent class method which has been overridden in the child class, from the child class.
- The super() in-built function can be used to call/refer the Parent class without explicitly naming them. This helps in situations where the Parent class name may change. Hence, super() helps in avoiding strong ties with class names and increases maintainability.
- super() helps the most when there are multiple inheritance happening, and the MRO ends up being complex. In case you need to call a method from a specific parent class, use super().
- There are multiple ways to call a method from a Parent class.
. - super(
, self). - super().