∙디자인패턴

[디자인패턴] 전략 패턴 사용하기

coor 2022. 11. 19. 01:17

들어가며

전략 패턴 정의는 각각의 알고리즘 군을 교환이 가능하도록 별도로 정의하고 각각 캡슐화한 후 서로 교환해서 사용할 수 있는 패턴이다. 이 패턴을 적용시키기 위해서 회원 정보와 결제 정보를 외부에 요청을 보내는 서비스에 적용해보았다. 또한 제네릭을 통해 각각 VO 객체를 보낼 수 있도록 유연함도 추가하였다. 큰 구조로 보면 요청을 보내는 인터페이스와 각각 전략을 넣을 수 있는 구현 클래스들로 구성된다.

코드 설계

GateRequestService.java

public interface GateRequestService<T> {
    void request(T object);
}
  • 외부에 요청을 보내는 메서드 생성
  • 제네릭 vo 객체를 받을 수 있도록 설정


GateRequestSignUpService.java
GateRequestOrderService.java

@Service("gateRequestSignUpService")
@RequiredArgsConstructor
public class GateRequestSignUpService implements GateRequestService<Account> {

    private final GateApiClient gateApiClient;

    @Override
    public void request(Account account) {

        // 회원 유효성 체크
        ...

        // 외부 요청
        gateApiClient.post(account);
    }
}


@Service("gateRequestOrderService")
@RequiredArgsConstructor
public class GateRequestOrderService implements GateRequestService<Orders> {

    private final GateApiClient gateApiClient;

    @Override
    public void request(Orders orders) {

        // 주문 유효성 체크
        ...

        // 외부 요청
        gateApiClient.post(orders);
    }
}
  • 각 구현 클래스의 빈을 식별하기 위해 카멜 케이스로 빈 이름 등록
  • GateRequestService.request 상속받은 후 제네릭 안에 객체 주입
  • 각각 전략에 맞게 유효성 체크(회원, 주문)
  • 외부에 요청할 수 있는 GateApiClient 주입 받은 후 post 요청


AccountController.java

@Controller
@RequiredArgsConstructor
public class AccountController {

    @Resource(name = "gateRequestSignUpService")
    private final GateRequestService<Account> gateRequestService;
    private final AccountService accountService;

    @PostMapping("/account/signup")
    public ResponseEntity<boolean> postSignup(@RequestBody Account account) {
        
        gateRequestService.request(account)
        
        return ResponseEntity.ok(accountService.postSignup(account));
    }
}


@Controller
@RequiredArgsConstructor
public class OrdersController {

    @Resource(name = "gateRequestOrdersService")
    private final GateRequestService<Orders> gateRequestService;
    private final OrdersService ordersSerivce;

    @PostMapping("/orders")
    public ResponseEntity<boolean> postOrders(@RequestBody Orders orders) {
        
        gateRequestService.request(orders)
        
        return ResponseEntity.ok(ordersSerivce.postOrders(orders));
    }
}
  • GateRequestService의 구현 클래스를 식별할 수 있게 @Resource 어노테이션을 통해 주입
    - GateRequestService -> 회원 데이터 -> GateRequestSignUpService
    - GateRequestService -> 주문 데이터 -> GateRequestOrderService
  • 각 구현 클래스의 vo 객체를 받을 수 있게 객체 주입



결론

인터페이스 목적인 어떤 요청이 왔을 때 그 요청이 필요한 서비스를 찾아주는 것으로 흔히 많이 쓰는 JPA의 Querydsl는 큰 Querydsl 안에 구현 클래스 중에 하나이다. 같은 맥락으로 큰 GateRequestService 인터페이스 안에 GateRequestSignupService와 GateRequestOrdersService로 나뉘게 된다. 이렇게 함으로써 OCP 원칙도 지킬 수 있으며 다른 외부 요청에 보내야 하는 상황이 오면 구현 클래스를 하나 더 생성을 통해 유연함을 가지게 된다.