5 SOLID Principles In TypeScript

By zooboole

In the world of software development, producing code that is not only functional but also maintainable, scalable, and adaptable is of paramount importance. The SOLID principles, a set of five design principles, provide guidelines for achieving these goals. These principles were introduced by Robert C. Martin and have since become a cornerstone of object-oriented programming. In this article, we will explore each of the SOLID principles and demonstrate their application using TypeScript.

1. Single Responsibility Principle (SRP)

The Single Responsibility Principle states that a class should have only one reason to change. In other words, a class should have only one responsibility or job. This ensures that the class remains focused and doesn't become bloated with unrelated functionality.

Let's see an example of SRP violation and its refactoring in TypeScript:

// SRP Violation
class User {
    constructor(private name: string) {}

    getName() {
        return this.name;
    }

    saveToDatabase() {
        // Code to save user to database
    }
}

In the above code, the User class has two responsibilities: managing user data and interacting with the database. This violates the SRP. Let's refactor it:

// SRP Applied
class User {
    constructor(private name: string) {}

    getName() {
        return this.name;
    }
}

class UserRepository {
    saveToDatabase(user: User) {
        // Code to save user to database
    }
}

By splitting the responsibilities into separate classes, we adhere to the SRP, making the code easier to maintain and extend.

2. Open/Closed Principle (OCP)

The Open/Closed Principle emphasizes that software entities (classes, modules, functions) should be open for extension but closed for modification. This means that you should be able to add new functionality without altering existing code.

Let's illustrate the OCP using a TypeScript example:

// OCP Violation
class Rectangle {
    constructor(public width: number, public height: number) {}
}

class AreaCalculator {
    calculateArea(shape: Rectangle) {
        return shape.width * shape.height;
    }
}

Here, the AreaCalculator is tightly coupled to the Rectangle class, making it hard to add new shapes without modifying the AreaCalculator class. Let's refactor it:

// OCP Applied
interface Shape {
    calculateArea(): number;
}

class Rectangle implements Shape {
    constructor(public width: number, public height: number) {}

    calculateArea() {
        return this.width * this.height;
    }
}

class Circle implements Shape {
    constructor(public radius: number) {}

    calculateArea() {
        return Math.PI * this.radius * this.radius;
    }
}

By using the Shape interface and creating individual shape classes, we've adhered to the OCP, allowing easy extension without altering existing code.

3. Liskov Substitution Principle (LSP)

The Liskov Substitution Principle states that objects of a superclass should be replaceable with objects of its subclasses without affecting the correctness of the program. In simpler terms, derived classes should be able to substitute their base classes seamlessly.

Let's demonstrate the LSP with TypeScript:

// LSP Violation
class Bird {
    fly() {
        // Code for flying
    }
}

class Ostrich extends Bird {
    // Ostriches can't fly
}

function makeBirdFly(bird: Bird) {
    bird.fly();
}

In this case, the Ostrich class inherits from Bird, but it doesn't support the fly behavior. This violates the LSP. Let's fix it:


// LSP Applied
interface Flyable {
    fly(): void;
}

class Bird implements Flyable {
    fly() {
        // Code for flying
    }
}

class Ostrich {
    // Ostrich-specific behavior
}

function makeAnimalFly(flyable: Flyable) {
    flyable.fly();
}

By separating the flying behavior into an interface and allowing only flyable objects to use it, we uphold the LSP.

4. Interface Segregation Principle (ISP)

The Interface Segregation Principle suggests that clients should not be forced to depend on interfaces they don't use. In other words, large interfaces should be split into smaller, more specific ones to avoid unnecessary dependencies.

Let's apply the ISP in TypeScript:

// ISP Violation
interface Worker {
    work(): void;
    eat(): void;
}

class OfficeWorker implements Worker {
    work() {
        // Code for working in the office
    }

    eat() {
        // Code for eating during break
    }
}

class FactoryWorker implements Worker {
    work() {
        // Code for working in the factory
    }

    eat() {
        // Code for eating during break
    }
}

Here, both OfficeWorker and FactoryWorker are forced to implement methods they don't need. Let's rectify this:


// ISP Applied
interface Workable {
    work(): void;
}

interface Eatable {
    eat(): void;
}

class OfficeWorker implements Workable, Eatable {
    work() {
        // Code for working in the office
    }

    eat() {
        // Code for eating during break
    }
}

class FactoryWorker implements Workable {
    work() {
        // Code for working in the factory
    }
}

By splitting the Worker interface into Workable and Eatable, we ensure that classes only implement the methods they require, adhering to the ISP.

5. Dependency Inversion Principle (DIP)

The Dependency Inversion Principle states that high-level modules should not depend on low-level modules; both should depend on abstractions. Furthermore, abstractions should not depend on details; details should depend on abstractions.

Let's see an example of DIP in TypeScript:

// DIP Violation
class LightBulb {
    turnOn() {
        // Code to turn on the light bulb
    }

    turnOff() {
        // Code to turn off the light bulb
    }
}

class Switch {
    private bulb = new LightBulb();

    flip() {
        if (/* condition */) {
            this.bulb.turnOn();
        } else {
            this.bulb.turnOff();
        }
    }
}

In this scenario, the Switch class directly depends on LightBulb, violating the DIP. Let's improve it:


// DIP Applied
interface Switchable {
    turnOn(): void;
    turnOff(): void;
}

class LightBulb implements Switchable {
    turnOn() {
        // Code to turn on the light bulb
    }

    turnOff() {
        // Code to turn off the light bulb
    }
}

class Switch {
    private device: Switchable;

    constructor(device: Switchable) {
        this.device = device;
    }

    flip() {
        if (/* condition */) {
            this.device.turnOn();
        } else {
            this.device.turnOff();
        }
    }
}

By introducing the Switchable interface, we've inverted the dependency direction, adhering to the DIP.

Conclusion

The SOLID principles provide a foundation for writing clean, maintainable, and flexible code. By understanding and applying these principles, developers can create systems that are more robust, easier to extend, and less prone to bugs. In this article, we explored each of the SOLID principles — Single Responsibility, Open/Closed, Liskov Substitution, Interface Segregation, and Dependency Inversion — and demonstrated their implementation using TypeScript. Incorporating these principles into your design and development process can significantly improve the quality and longevity of your software projects.

Last updated 2023-08-09 UTC