웹 백엔드 관점으로 본 디자인 패턴 정리(2) - 전략 패턴(Strategy Pattern)
앞으로 패턴들에 대해서 어떻게 백엔드의 관점에서 사용할 것이며,
얼마나 유용할지 평가를 해보려고 한다.
애초에 패턴들을 실용적인 관점에서 검토하지 않으면 의미가 없다는 생각이다.
이 공부가 나의 실무 코드에 도움이 되어야하니 그런 점들을 위해서 정리해두려고 한다.
그러나 딱 내 학습 수준에 맞는 평가가 나올 것 같아서 그 점은 참고해야 한다.
추가로 디자인 패턴의 개념에 대한 기초 설명은 패스하려고 한다.
전략 패턴
- 필요 개념: 컴포지션
- 활용도: 부담없이 사용해볼만 함
- 난이도: 간단
- 패턴이 필요한 상황: 상속 상황에서 내부 코드의 빈번한 중복
보통 이 패턴에 가장 많이 나오는 예제는 Duck에 대한 예제이다.
quack, fly 메서드가 나오고 여러 오리종류가 나오는 그 예제다.
헤드퍼스트에서 예로 들어서 유명하지만,
솔직히 이런 예시는 기초를 이해할 때나 도움이 되지 실제 프로그램과는 너무 동떨어진다.
백엔드 어플리케이션을 개발하면서 상속이 많아지는 순간이 종종 존재하곤 한다.
스프링을 기준으로 보통 두 레이어에서 상속이 많이 발생한다.
1. 서비스: @Service 어노테이션을 달고 있는 스프링 빈
2. 도메인 객체: 다양하기는 하지만 JPA 기준 @Entity가 달린 객체가 대표적이다.
서비스에서는 공통의 상위 클래스가 있고 다양한 요구사항에 따라서 하위 서비스가 생성되는 식이다.
결제를 예로 들면 이런 식으로 구성이 될 수가 있을 것 같다.
public interface PaymentService { ... }
public class CardPaymentService implements PaymentService { ... }
public class MobilePaymentService implements PaymentService { ... }
public class KakaoPaymentService implements PaymentService { ... }
과연 이게 끝일까?
비즈니스 요구사항은 변화무쌍해서 여기서 끝나지 않을 가능성이 무조건 백퍼센트다.
ios, google이 추가될 수도 있고, 다른 결제수단이 추가될 수도 있고,
같은 결제수단인데 살짝 다른 서비스가 필요할 수도 있다.
일단 클래스가 늘어나는 걸 막을 수 없는 건 확정이다.
일단 처음에 클래스가 몇 종류 되지 않을 때는 복붙으로 개발하는 게 맞다.
그러다가 슬슬 중복이 보인다 싶은 때가 오는데 그때가 전략패턴이 슬슬 도입될 때이다.
@Service
public class CardPaymentServiceA implements PaymentService{
public void startPayment(){
... 수십줄의 로직 A ...
}
public void endPayment(){
... 수십줄의 로직 B ...
}
public void cancelPayment(){
... 수십줄의 로직 C ...
}
}
@Service
public class CardPaymentServiceB implements PaymentService{
public void startPayment(){
... 수십줄의 로직 A ...
}
public void endPayment(){
... 수십줄의 로직 B ...
}
public void cancelPayment(){
... 수십줄의 로직 D ...
}
}
정말 싫은 상황이지만 어쩌다보니 거의 다 비슷한데 약간 비슷한 로직의 카드결제가 하나가 더 추가되었는데,
(이런 일은 정말 일어날 수 있다) 딱 cancelPayment만 코드가 다르다.
그럼 당연히 CardPaymentService A와 B의 중복을 빼내야 할 것이고 그 때 전략패턴을 사용하면 된다.
@Service
public class CardPaymentServiceA implements PaymentService{
private StartPaymentBehavior spb;
private EndPaymentBehavior epb;
private CancelPaymentBehavior cpb;
public CardPaymentServiceA(){
this.spb = new startMethodA();
this.epb = new endMethodB();
this.cpb = new cancelMethodC();
}
public void startPayment(){
spb.start();
}
public void endPayment(){
epb.end();
}
public void cancelPayment(){
cpb.cancel();
}
}
아마도 CardPaymentServiceB에서는
this.cpb = new cancelMethodD();
가 사용될 것으로 예상할 수 있다.
또한 전략 패턴을 통해서 유닛 테스트를 하기도 좀 더 편해진 것 같다.
원래대로였으면 각 서비스별로 테스트 코드를 다 만들어야 했지만,
이렇게 중복 코드를 모아두면 각 전략 클래스만 테스트할 수 있어서 편리하다.
추가로 상속 때문에 복잡해진 코드를 컴포지션을 이용해서 간단하게 만들었다는 점을 기억하면 좋다.
이 패턴에 대해서 정리하자면
'상속이 많아져서 중복이 겉잡을 수 없어져 복붙이 많이 필요한 상황이라면 도입을 고려해볼만 하다.'
애초에 협업하는 사람들이 따라가기 어려운 부분도 적어서 부담없이 도입해볼만 한 것 같다.
다만 한두개의 상속 케이스로 이 패턴을 도입하는 건 좀 아닌 것 같다.