개발 언어/기타 웹개발 지식

SOLID 원칙: 객체 지향 프로그래밍의 기반

jjiiiinn 2024. 8. 14. 22:35
728x90

SOLID란 무엇인가?

객체 지향 프로그래밍에서 SOLID는 소프트웨어 설계의 이해, 개발, 유지보수를 향상시키기 위한 다섯 가지 설계 원칙의 약어입니다.

이 원칙들을 적용하면 버그 감소, 코드 품질 향상, 더 체계적인 코드 생산, 결합도 감소, 리팩토링 개선, 코드 재사용 촉진 등의 효과를 볼 수 있습니다.

 

SOLID 원칙 깊게 살펴보기

1. 단일 책임 원칙 (SRP: Single Responsibility Principle)

핵심: 한 클래스는 하나의, 오직 하나의 변경 이유만 가져야 한다.

왜 중요한가?

  • 클래스의 응집도 향상
  • 유지보수 용이성 증가
  • 다른 클래스와의 결합도 감소

예시:

// 잘못된 예
public class ProfileManager {
    public boolean authenticateUser(String username, String password) {
        // 인증 로직
    }

    public UserProfile showUserProfile(String username) {
        // 사용자 프로필 표시 로직
    }

    public UserProfile updateUserProfile(String username) {
        // 사용자 프로필 업데이트 로직
    }

    public void setUserPermissions(String username) {
        // 권한 설정 로직
    }
}

// 개선된 예
public class AuthenticationManager {
    public boolean authenticateUser(String username, String password) {
        // 인증 로직
    }
}

public class UserProfileManager {
    public UserProfile showUserProfile(String username) {
        // 사용자 프로필 표시 로직
    }

    public UserProfile updateUserProfile(String username) {
        // 사용자 프로필 업데이트 로직
    }
}

public class PermissionManager {
    public void setUserPermissions(String username) {
        // 권한 설정 로직
    }
}

2. 개방-폐쇄 원칙 (OCP: Open-Closed Principle)

핵심: 객체나 개체는 확장에는 열려 있어야 하지만, 수정에는 닫혀 있어야 한다.

장점:

  • 기존 코드 수정 없이 새로운 기능 추가 가능
  • 코드의 유연성과 재사용성 향상
  • 시스템 안정성 개선

예시:

// OCP를 적용하기 전
public class Circle {
    public double radius;
}

public class Square {
    public double side;
}

public class AreaCalculator {
    public double totalArea(Object[] shapes) {
        double area = 0;
        for (Object shape : shapes) {
            if (shape instanceof Circle) {
                Circle circle = (Circle) shape;
                area += Math.PI * circle.radius * circle.radius;
            } else if (shape instanceof Square) {
                Square square = (Square) shape;
                area += square.side * square.side;
            }
        }
        return area;
    }
}

// OCP를 적용한 후
public interface Shape {
    double area();
}

public class Circle implements Shape {
    public double radius;

    @Override
    public double area() {
        return Math.PI * radius * radius;
    }
}

public class Square implements Shape {
    public double side;

    @Override
    public double area() {
        return side * side;
    }
}

public class AreaCalculator {
    public double totalArea(Shape[] shapes) {
        double area = 0;
        for (Shape shape : shapes) {
            area += shape.area();
        }
        return area;
    }
}

3. 리스코프 치환 원칙 (LSP: Liskov Substitution Principle)

핵심: 파생 클래스는 기본 클래스를 대체할 수 있어야 한다.

중요성:

  • 다형성과 확장성의 올바른 사용 보장
  • 예측 가능한 코드 동작 유지
  • 코드 재사용성 증가

예시:

class Person {
    public String speakName() {
        return "I am a person!";
    }
}

class Child extends Person {
    @Override
    public String speakName() {
        return "I am a child!";
    }
}

public class Main {
    public static void printName(Person person) {
        System.out.println(person.speakName());
    }

    public static void main(String[] args) {
        Person person = new Person();
        Child child = new Child();

        printName(person); // 출력: I am a person!
        printName(child);  // 출력: I am a child!
    }
}

4. 인터페이스 분리 원칙 (ISP: Interface Segregation Principle)

핵심: 클래스는 사용하지 않는 인터페이스를 구현하도록 강제되어서는 안 된다.

장점:

  • 불필요한 의존성 제거
  • 인터페이스의 응집도 향상
  • 코드의 유연성과 재사용성 증가

예시:

// ISP를 위반한 예
interface Book {
    void read();
    void download();
}

class OnlineBook implements Book {
    public void read() {
        // 읽기 로직
    }

    public void download() {
        // 다운로드 로직
    }
}

class PhysicalBook implements Book {
    public void read() {
        // 읽기 로직
    }

    public void download() {
        // 물리적 책에는 의미 없는 메서드
        throw new UnsupportedOperationException();
    }
}

// ISP를 적용한 예
interface Readable {
    void read();
}

interface Downloadable {
    void download();
}

class OnlineBook implements Readable, Downloadable {
    public void read() {
        // 읽기 로직
    }

    public void download() {
        // 다운로드 로직
    }
}

class PhysicalBook implements Readable {
    public void read() {
        // 읽기 로직
    }
}

5. 의존관계 역전 원칙 (DIP: Dependency Inversion Principle)

핵심: 추상화에 의존해야 하며, 구체화에 의존하면 안 된다.

상위 모듈은 하위 모듈에 의존해서는 안 된다. 둘 다 추상화에 의존해야 한다.

추상화는 세부 사항에 의존해서는 안 된다. 세부사항이 추상화에 의존해야 한다.

 

예시:

// DIP를 적용하기 전
class MySQLDatabase {
    public String getUserData(int id) {
        // MySQL 데이터베이스에서 사용자 데이터를 가져오는 로직
    }
}

class UserService {
    private MySQLDatabase database;

    public UserService() {
        this.database = new MySQLDatabase();
    }

    public String getUser(int id) {
        return this.database.getUserData(id);
    }
}

// DIP를 적용한 후
interface Database {
    String getUserData(int id);
}

class MySQLDatabase implements Database {
    public String getUserData(int id) {
        // MySQL 데이터베이스에서 사용자 데이터를 가져오는 로직
    }
}

class PostgreSQLDatabase implements Database {
    public String getUserData(int id) {
        // PostgreSQL 데이터베이스에서 사용자 데이터를 가져오는 로직
    }
}

class UserService {
    private Database database;

    public UserService(Database database) {
        this.database = database;
    }

    public String getUser(int id) {
        return this.database.getUserData(id);
    }
}

결론

SOLID 원칙은 객체 지향 프로그래밍에서 깔끔하고 유지보수가 쉬운 코드를 작성하기 위한 강력한 가이드라인입니다. 이 원칙들을 이해하고 적용함으로써, 우리는 더 나은 소프트웨어 설계와 개발을 할 수 있습니다. 물론 실제 상황에서는 이 원칙들을 완벽하게 적용하기 어려울 수 있지만, 이를 지향점으로 삼고 노력한다면 분명 더 좋은 코드를 작성할 수 있을 것입니다.

참고 블로그

https://dev.to/lukeskw/solid-principles-theyre-rock-solid-for-good-reason-31hn

728x90