Object-Oriented Programming (OOP) is a key approach used in software development.

In this article, we'll explore the main ideas of OOP, particularly looking at classes, objects, inheritance, and polymorphism in Python.

By the end of this guide, you'll understand how to organize your Python code using OOP principles, making your programs more modular, reusable, and easier to maintain.

If you prefer you can follow along with the video version:


What is Object-Oriented Programming?

Object-Oriented Programming (OOP) organizes software design around data, or objects, rather than functions and logic.

An object is like a container with unique attributes (data) and behaviors (functions). OOP focuses on several key concepts:

Encapsulation

This means bundling the data (attributes) and the methods (functions) that operate on that data into a single unit, called a class.

It also involves restricting access to some components of the object, making it more secure.

Abstraction

This is the idea of hiding the complex implementation details and showing only the essential features of the object.

It reduces complexity and allows the programmer to focus on higher-level interactions.

Inheritance

This is a mechanism for creating a new class (derived class) from an existing class (base class).

The new class inherits attributes and methods from the existing class.

Polymorphism

This is the ability to use a single interface to represent different data types.

It allows objects to be treated as instances of their parent class and makes it possible to define methods in a child class that have the same name as a method in the parent class.


Are you tired of writing the same old Python code? Want to take your programming skills to the next level? Look no further! This book is the ultimate resource for beginners and experienced Python developers alike.

Get "Python's Magic Methods - Beyond __init__ and __str__"

Magic methods are not just syntactic sugar, they're powerful tools that can significantly improve the functionality and performance of your code. With this book, you'll learn how to use these tools correctly and unlock the full potential of Python.

OOP Basics in Python: Classes and Objects

At the core of Object-Oriented Programming (OOP) in Python are classes and objects.

Classes

A class is like a blueprint for creating objects.

It defines a set of properties (attributes) and actions (methods) that the objects will have.

In Python, you create a class using the class keyword. Here's an example:

class Car:
    def __init__(self, make, model, year):
        self.make = make
        self.model = model
        self.year = year

    def start_engine(self):
        print(f"{self.make} {self.model}'s engine started.")

Objects

An object is an instance of a class.

Once you define a class, you can create multiple objects (instances) from it.

Each object can have its own unique values for the attributes defined in the class.

Here's how you create and use an object:

my_car = Car("Toyota", "Corolla", 2020)
my_car.start_engine()  # Output: Toyota Corolla's engine started.

In this example, my_car is an object of the Car class.

It has its own values for make, model, and year, and you can use methods like start_engine.


Inheritance in Python

Inheritance lets one class (the child class) take on the attributes and methods of another class (the parent class).

This is great for reusing code and setting up a hierarchy between classes.

Here's an example:

class Vehicle:
    def __init__(self, make, model):
        self.make = make
        self.model = model

    def drive(self):
        print("Driving...")


class Car(Vehicle):
    def __init__(self, make, model, year):
        super().__init__(make, model)
        self.year = year

    def start_engine(self):
        print(f"{self.make} {self.model}'s engine started.")


my_car = Car("Honda", "Civic", 2021)
my_car.drive()  # Output: Driving...
my_car.start_engine()  # Output: Honda Civic's engine started.

In this example, the Car class inherits from the Vehicle class.

Because of this, the Car class can use the drive method that's defined in the Vehicle class.

Method Overriding

Sometimes, a child class needs to change or add to the behavior of a method it inherits from a parent class.

This is done through method overriding.

Here's an example:

class Vehicle:
    def drive(self):
        print("Driving a vehicle...")


class Car(Vehicle):
    def drive(self):
        print("Driving a car...")


my_vehicle = Vehicle()
my_vehicle.drive()  # Output: Driving a vehicle...

my_car = Car()
my_car.drive()  # Output: Driving a car...

In this example, the drive method in the Car class overrides the drive method in the Vehicle class, allowing for customized behavior.

Multiple Inheritance

Python also supports multiple inheritance, where a class can inherit from more than one base class.

Here's an example:

class Vehicle:
    def __init__(self, make, model):
        self.make = make
        self.model = model

    def drive(self):
        print("Driving a vehicle...")


class Electric:
    def charge(self):
        print("Charging...")


class Car(Vehicle):
    def __init__(self, make, model, year):
        super().__init__(make, model)
        self.year = year

    def start_engine(self):
        print(f"{self.make} {self.model}'s engine started.")


class HybridCar(Car, Electric):
    def switch_mode(self):
        print("Switching to electric mode...")


my_hybrid = HybridCar("Toyota", "Prius", 2022)
my_hybrid.start_engine()  # Output: Toyota Prius's engine started.
my_hybrid.drive()  # Output: Driving a vehicle...
my_hybrid.charge()  # Output: Charging...
my_hybrid.switch_mode()  # Output: Switching to electric mode...

In this example, the HybridCar class inherits from both Car and Electric, allowing it to access methods from both parent classes.


Polymorphism in Python

Polymorphism is a feature that allows methods to perform different actions based on the object they are working with, even if these methods have the same name.

This is especially useful when dealing with inheritance, as it lets you use the same method name across different classes in a way that makes sense for each class.

Polymorphism with Functions

Here's an example:

class Dog:
    def speak(self):
        return "Woof!"


class Cat:
    def speak(self):
        return "Meow!"


def make_animal_speak(animal):
    print(animal.speak())


dog = Dog()
cat = Cat()

make_animal_speak(dog)  # Output: Woof!
make_animal_speak(cat)  # Output: Meow!

The make_animal_speak function demonstrates polymorphism by accepting any object with a speak method.

This allows it to work with both Dog and Cat objects, despite their differences.

Polymorphism with Class Methods

Polymorphism also comes into play when working with methods in a class hierarchy.

Here's an example:

class Animal:
    def speak(self):
        raise NotImplementedError("Subclass must implement abstract method")


class Dog(Animal):
    def speak(self):
        return "Woof!"


class Cat(Animal):
    def speak(self):
        return "Meow!"


animals = [Dog(), Cat()]

for animal in animals:
    print(animal.speak())

In this example, both Dog and Cat are subclasses of Animal.

The speak method is implemented in both subclasses, allowing polymorphism to take effect when iterating through the list of animals.


Encapsulation and Data Hiding

Encapsulation is the practice of combining data and the methods that work on that data into a single unit, called a class.

It also involves restricting access to certain parts of the object, which is crucial for protecting data in Object-Oriented Programming (OOP).

Private and Public Attributes

In Python, you can indicate that an attribute is private by starting its name with an underscore.

While this doesn't actually prevent access from outside the class, it's a convention that signals that the attribute should not be accessed directly.

Here's an example:

class Account:
    def __init__(self, owner, balance=0):
        self.owner = owner
        self._balance = balance  # Private attribute

    def deposit(self, amount):
        self._balance += amount

    def withdraw(self, amount):
        if amount <= self._balance:
            self._balance -= amount
        else:
            print("Insufficient funds")

    def get_balance(self):
        return self._balance


my_account = Account("John", 1000)
my_account.deposit(500)
print(my_account.get_balance())  # Output: 1500

In this example, the Account class has a private attribute _balance, which is manipulated through methods like deposit, withdraw, and get_balance.

Direct access to _balance from outside the class is discouraged.


Advanced OOP Concepts

For those who want to deepen their understanding of Object-Oriented Programming (OOP) in Python, here are a few advanced topics:

Class Methods

These are methods that are connected to the class itself, not to individual instances of the class.

They can change the state of the class, which affects all instances of the class.

class Car:
    total_cars = 0

    def __init__(self, make, model):
        self.make = make
        self.model = model
        Car.total_cars += 1

    @classmethod
    def get_total_cars(cls):
        return cls.total_cars

Static Methods

These are methods that belong to the class but do not change the state of the class or its instances.

They are defined using the @staticmethod decorator.

class MathOperations:
    @staticmethod
    def add(x, y):
        return x + y

Property Decorators

Property decorators in Python provide a way to define getters, setters, and deleters for class attributes in a more Pythonic manner.

class Employee:
    def __init__(self, name, salary):
        self._name = name
        self._salary = salary

    @property
    def salary(self):
        return self._salary

    @salary.setter
    def salary(self, value):
        if value < 0:
            raise ValueError("Salary cannot be negative")
        self._salary = value

In this example, the salary attribute is accessed like a regular attribute but is managed by getter and setter methods.


Conclusion

Object-Oriented Programming (OOP) in Python is a powerful way to organize and manage your code.

By learning the principles of OOP, such as classes, objects, inheritance, polymorphism, and encapsulation, you can write Python programs that are well-organized, reusable, and easy to maintain.

Whether you're working on small scripts or large applications, using OOP principles will help you create more efficient, scalable, and robust software.


If you would like this post, learn from it, and would like to support my work, consider leaving a small donation:


If you are not a Premium Django Series Subscriber, sign up below and get a 25% discount on the yearly membership:

Feel free to use the feedback buttons or send in a comment for future content you would love to see or what new things you learned from this post.

Also, share it with a friend who would love to read it.


Tagged in: