오봉이와 함께하는 개발 블로그

스프링 핵심 원리 - 고급편 > 전략 패턴 (시작, 예제 1) 본문

BE/Spring

스프링 핵심 원리 - 고급편 > 전략 패턴 (시작, 예제 1)

오봉봉이 2024. 5. 10. 00:18
728x90

전략 패턴 - 시작

  • ContextV1Test
@Slf4j
public class ContextV1Test {

  @Test
  void strategy() {
    logic1();
    logic2();
  }

  private void logic1() {
    long startTime = System.currentTimeMillis(); //비즈니스 로직 실행
    log.info("비즈니스 로직1 실행");
    //비즈니스 로직 종료
    long endTime = System.currentTimeMillis();
    long resultTime = endTime - startTime;
    log.info("resultTime={}", resultTime);
  }
  private void logic2() {
    long startTime = System.currentTimeMillis(); //비즈니스 로직 실행
    log.info("비즈니스 로직2 실행");
    //비즈니스 로직 종료
    long endTime = System.currentTimeMillis();
    long resultTime = endTime - startTime;
    log.info("resultTime={}", resultTime);
  }
}

전략 패턴 - 예제 1

탬플릿 메서드 패턴은 부모 클래스에 변하지 않는 템플릿을 두고, 변하는 부분을 자식 클래스에 두어서 상속을 사용해서 문제를 해결했다.
전략 패턴은 변하지 않는 부분을 Context라는 곳에 두고, 변하는 부분을 Strategy라는 인터페이스를 만들고 해당 인터페이스를 구현하도록 해서 문제를 해결한다.
상속이 아니라 위임으로 문제를 해결하는 것이다.
전략 패턴에서 Context는 변하지 않는 템플릿 역할을 하고, Strategy는 변하는 알고리즘 역할을 한다.

GOF 디자인 패턴에서 정의한 전략 패턴의 의도는 다음과 같다.

알고리즘 제품군을 정의하고 각각을 캡슐화하여 상호 교환 가능하게 만들자. 전략을 사용하면 알고리즘을 사용하는 클라이언트와 독립적으로 알고리즘을 변경할 수 있다.

image

Strategy 인터페이스

public interface Strategy {
  void call();
}

ContextV1

@Slf4j
public class ContextV1 {

  private Strategy strategy;

  public ContextV1(Strategy strategy) {
    this.strategy = strategy;
  }

  public void execute() {
    long startTime = System.currentTimeMillis();
    //비즈니스 로직 실행
    strategy.call(); //위임
    //비즈니스 로직 종료
    long endTime = System.currentTimeMillis();
    long resultTime = endTime - startTime;
    log.info("resultTime={}", resultTime);
  }
}

ContextV1은 변하지 않는 로직을 가지고 있는 템플릿 역할을 하는 코드다.
전략 패턴에서는 이것을 컨텍스트라 한다.
쉽게 이야기해서 컨텍스트는 크게 변하지 않지만, 그 문맥 속에서 strategy를 통해 일부 전략이 변경된다.

Context는 내부에 Strategy strategy 필드를 가지고 있고, 생성자를 통해 이 Strategy 인터페이스의 구현체를 주입하면 된다.
전략 패턴의 핵심은 Context는 Strategy에만 의존한다는 점이다. 때문에 Strategy의 구현체를 변경하거나, 새로 만들어도 Context에는 영향을 주지 않으며 변경 범위 또한 해당 Strategy 구현체를 사용하는 곳에만 발생한다.

ContextV1Test 추가

  @Test
  void strategyV1() {
    Strategy strategyLogic1 = new StrategyLogic1();
    ContextV1 context1 = new ContextV1(strategyLogic1);
    context1.execute();

    Strategy strategyLogic2 = new StrategyLogic2();
    ContextV1 context2 = new ContextV1(strategyLogic2);
    context2.execute();
  }

전략 패턴 도식

image

  1. Context에 원하는 Strategy 구현체를 주입한다.
  2. 클라이언트는 context를 실행한다.
  3. contextcontext로직을 시작한다.
  4. context 로직 중간에 strategy.call()을 호출해서 주입 받은 strategy 로직을 실행한다.
  5. context는 나머지 로직을 실행한다.

 

출처: 김영한 지식공유자님의 스프링 핵심 원리 고급편

728x90
Comments