DEV Community

Cover image for Object-Oriented Concepts: The Foundation of Modern Programming - Part 01
Kaweesha Marasinghe
Kaweesha Marasinghe

Posted on

Object-Oriented Concepts: The Foundation of Modern Programming - Part 01

What Came Before OOP

Procedural Programming (1950s-70s)

Procedural programming was a paradigm that followed a step-by-step approach to solving problems by breaking them into smaller tasks or procedures.

In its early stages, C was a widely used procedural programming language. Here’s an example of a simple procedural program in C:

#include <stdio.h>

void calculateTotal() {
    int price1 = 10, price2 = 20, total;
    total = price1 + price2;
    printf("Total: %d\n", total);
}

int main() {
    calculateTotal();
    return 0;
}
Enter fullscreen mode Exit fullscreen mode

While procedural programming worked well for smaller, straightforward programs, it started to face challenges as software became more complex. This complexity brought several issues:

  1. Code Duplication: The same logic was often repeated in different parts of the program.
  2. Maintenance Problems: A single change required modifications in multiple places, increasing the risk of errors.
  3. Scalability Issues: Procedural programming struggled to adapt to the growing demands of larger, more complex software.
  4. Poor Representation of Real-World Scenarios: Data and operations were kept separate, making it difficult to model real-world entities like a "Bank Account" or "Car."

The Shift to Object-Oriented Programming (OOP)

As software complexity grew, procedural programming began to show its limitations. This led to the evolution of a new paradigm—Object-Oriented Programming (OOP). OOP revolutionized the way we design and build software by modeling real-world entities and combining data and behavior into a single unit called an object.

Key Concepts of OOP

1. Encapsulation 🛡

Encapsulation binds data (attributes) and behaviors (functions) that operate on the data into a single entity, typically a class. This means we can group related data and methods together into one unit and control how the data is accessed and modified. This promotes data security and modularity.

Encapsulation

Encapsulation is implemented with the support of access specifiers:

  • Private variables: These protect sensitive data by restricting outside access to variables.
  • Public methods: These provide functionality to modify or access the private data from the outside.
Why Encapsulation is Important:
  • Data Protection: By using private access specifiers, some attributes (e.g., accBalance) are restricted from direct modification.
  • Modularity: Changes made internally won't affect external code that interacts with the object.
Example:

A great example of encapsulation is a TV remote.

The remote has several buttons that perform actions. Each button is coded to execute a related function when pressed, but you cannot change the underlying code that defines what each button does. The remote hides the implementation details, providing you with public methods (the buttons) while keeping the internal functions (private attributes) hidden.


2. Abstraction 🔍

As the complexity of a system grows, abstraction plays a vital role in managing that complexity. Abstraction is the concept that hides the complex details and only exposes the essential features.

The goal of abstraction is to simplify the interface and reduce the complexity for the user or objects interacting with it.

Example:

When you drive a car, you use the steering wheel, accelerator, brakes, and gear shifts, which are essential for driving.

But to drive, you don’t need to know how the gears are shifted, how the brake system works, or how the wheels rotate when you turn the steering wheel. These details are hidden (abstracted) from you.

You only interact with the simplified interface: the brake, accelerator, and steering wheel.


How to Use Abstraction in OOP

Abstract Classes:

Use abstract classes when you want to share common behavior but allow subclasses to provide specific implementations.

Example: A Shape class might have a method calculateArea(). You can abstract the calculation in the base class and allow subclasses like Circle, Rectangle, etc., to implement their specific area calculations.

Interfaces:

Use interfaces when you want to ensure a class implements certain behaviors but don’t care about the implementation details.

Example: An IPlayable interface that requires play() and pause() methods. Any class that implements this interface, like AudioPlayer or VideoPlayer, must provide implementations for play() and pause().


Code Example

// Abstract class - Shape (provides a blueprint for all shapes)
abstract class Shape {
    // Abstract method (does not have implementation)
    public abstract double calculateArea(); 

    // Concrete method (implemented in the abstract class)
    public void displayShape() {
        System.out.println("This is a shape.");
    }
}

// Concrete class - Circle (specific implementation of calculateArea)
class Circle extends Shape {
    private double radius;

    public Circle(double radius) {
        this.radius = radius;
    }

    @Override
    public double calculateArea() {
        return Math.PI * radius * radius;  // Area of circle
    }
}

// Concrete class - Rectangle (specific implementation of calculateArea)
class Rectangle extends Shape {
    private double length;
    private double width;

    public Rectangle(double length, double width) {
        this.length = length;
        this.width = width;
    }

    @Override
    public double calculateArea() {
        return length * width;  // Area of rectangle
    }
}

public class Main {
    public static void main(String[] args) {
        // Create objects of Circle and Rectangle
        Shape myCircle = new Circle(5.0);  // radius = 5
        Shape myRectangle = new Rectangle(4.0, 6.0);  // length = 4, width = 6

        // Display the shape and calculate areas
        myCircle.displayShape();
        System.out.println("Circle Area: " + myCircle.calculateArea());

        myRectangle.displayShape();
        System.out.println("Rectangle Area: " + myRectangle.calculateArea());
    }
}
Enter fullscreen mode Exit fullscreen mode

🌟 In Conclusion 🌟

As we've explored, the shift from procedural programming to Object-Oriented Programming (OOP) introduced us to powerful concepts such as Encapsulation and Abstraction that make software more manageable, scalable, and modular. 🌍💻

By focusing on objects that represent real-world entities, OOP allows developers to write cleaner and more maintainable code. 🎯

Embrace OOP, and start designing software that mirrors the complexity of the real world in an elegant and efficient manner! ✨


Top comments (0)