Objective

In this unit, we will explore the concept of inheritance in Object-Oriented Programming (OOP). Inheritance is a powerful feature that allows a class to inherit attributes and methods from another class. By the end of this unit, you will understand how to create subclasses, override methods, and use the super() function in Python.

Understanding Inheritance

Inheritance is a mechanism in OOP that allows a class to inherit attributes and methods from another class. This promotes code reusability and establishes a relationship between the parent (or base) class and the child (or derived) class. The child class can inherit attributes and methods from the parent class and can also have additional attributes and methods or override the existing ones.

Creating Subclasses

A subclass is a class that inherits from another class. You can create a subclass by passing the parent class as a parameter when defining the child class.

Here's an example:

class Shape:
    def describe(self):
        print("This is a shape.")

class Square(Shape):
    def describe(self):
        print("This is a square.")

In this example, Square is a subclass of Shape. It inherits the describe method from Shape but also overrides it with its own implementation.

Overriding Methods

A subclass can provide a specific implementation of a method that is already defined in its parent class. This is known as overriding. When the method is called on an object of the subclass, the overridden method in the subclass is executed, not the one in the parent class.

The super() Function

The super() function is used in a subclass to call a method from the parent class. This can be useful when you want to extend the behavior of a method in the parent class rather than completely overriding it.

Here's an example:

class Shape:
    def describe(self):
        print("This is a shape.")

class Square(Shape):
    def describe(self):
        super().describe()
        print("More specifically, this is a square.")

When you call describe on a Square object, it will first call the describe method in Shape, then print the additional message.

Project: Create Subclasses of the Shape Class that Represent Specific Shapes and Draw Them Using Turtle

In this project, you'll create subclasses for different shapes and use the Turtle library to draw them.

import turtle

# Define a base Shape class
class Shape:
    # The constructor takes three arguments
    def __init__(self, turtle, sides, length):
        self.t = turtle         # turtle instance
        self.sides = sides
        self.length = length
        
    def draw(self):
        # We don't want to implement this method in the Base class
        # so we use `pass` as a placeholder which means "do nothing".
        pass 

class Square(Shape):
    # The constructor takes two arguments
    def __init__(self, turtle, length):
        # Call super() on the parent object, specifing the number of sides
        super().__init__(turtle, 4, length)
        
    # Implement the draw method for a square
    def draw(self):
        for _ in range(self.sides):
            self.t.forward(self.length)
            self.t.right(90)

class Triangle(Shape):
    # The constructor takes two arguments
    def __init__(self, turtle, length):
        # Call super() on the parent object, specifing the number of sides
        super().__init__(turtle, 3, length)
        
    # Implement the draw method for a triangle
    def draw(self):
        for _ in range(self.sides):
            self.t.forward(self.length)
            self.t.right(120)

# Create a new turtle screen and set its background color
screen = turtle.Screen()
screen.setup(width=800, height=600)
screen.bgcolor("white")

# Initialize a turtle object
t = turtle.Turtle()

# Create Shape objects
shapes = [Square(t, 100), Triangle(t, 100)]

# Draw the shapes
for shape in shapes:
    shape.draw()
    t.penup()
    t.forward(150)
    t.pendown()

# Hide the turtle and wait until the window is closed
t.hideturtle()
turtle.done()

In this code, we have a parent class Shape with a method draw that does nothing. We then create two subclasses, Square and Triangle, each of which overrides the constructor __init__ method and the draw method to draw the appropriate shape using the Turtle library.

We create a list of shape objects and then use a loop to draw each shape in turn, moving the turtle to a new position between each shape to ensure they don't overlap.

This project demonstrates how inheritance can be used to create a flexible and reusable structure for representing and drawing different shapes. By defining a common interface in the parent class (Shape), we can easily add more shape subclasses in the future without changing the code that draws the shapes.

In the next unit, we'll continue exploring object-oriented programming, focusing on encapsulation and polymorphism.