Bami 2022. 10. 17. 14:33
728x90
반응형

Command 패턴

명령을 처리하는 객체를 통해 메서드와 실행되는 동작의 결합도를 낮출 수 있다

 

📜 원문: patterns.dev - command pattern

📜 번역: https://patterns-dev-kr.github.io/design-patterns/command-pattern/

 

커맨드 패턴을 사용하면 특정 작업을 실행하는 개체와 메서드를 호출하는 개체를 분리할 수 있습니다.

 

예를 들어 온라인 음식 배달 플랫폼을 개발한다고 가정해 보죠.

사용자는 주문하거나, 주문한 음식이 어디쯤 왔는지 확인하거나, 주문을 취소할 수 있습니다.

 

class OrderManager() {
  constructor() {
    this.orders = []
  }

  placeOrder(order, id) {
    this.orders.push(id)
    return `You have successfully ordered ${order} (${id})`;
  }

  trackOrder(id) {
    return `Your order ${id} will arrive in 20 minutes.`
  }

  cancelOrder(id) {
    this.orders = this.orders.filter(order => order.id !== id)
    return `You have canceled your order ${id}`
  }
}

 

OrderManager클래스에서 placeOrder, trackOrder, cancelOrder를 구현하였고, 이 메서드들을 직접 사용하는것도 가능합니다.

const manager = new OrderManager()

manager.placeOrder('Pad Thai', '1234')
manager.trackOrder('1234')
manager.cancelOrder('1234')

하지만 이렇게 manager의 메서드를 직접 사용하는 도중 나중에 특정 메서드의 이름을 변경하거나 메서드의 기능을 변경해야 하는 경우가 생길 수 있습니다.

 

placeOrder대신에 이름을 addOrder로 변경하면 전체 코드베이스에서 해당 메서드를 호출하지 않도록 코드를 수정해야 한다. 앱의 규모가 크면 까다로운 작업이 될 것입니다.

 

이렇게 하는 대신, manager 객체로부터 메서드를 분리하고 각각의 명령을 처리하는 함수를 만들것입니다.

 

먼저 OrderManager 클래스를 리펙토링 해 보죠. placeOrder, trackOrder, cancelOrder를 직집 구현하는 대신 

execute라는 하나의 메서드만 가지도록 합니다. 이 메서드는 인자로 주어진 어떤 명령이든 실행할 수 있습니다.

 

각 명령은 첫 번째 인자로 OrderManager의 orders 배열을 넘겨 접근할 수 있도록 합니다.

 

class OrderManager {
  constructor() {
    this.orders = []
  }

  execute(command, ...args) {
    return command.execute(this.orders, ...args)
  }
}

아래에서 3개의 Command 를 만듭니다.

  • PlaceOrderCommand
  • CancelOrderCommand
  • TrackOrderCommand
class Command {
  constructor(execute) {
    this.execute = execute
  }
}

function PlaceOrderCommand(order, id) {
  return new Command(orders => {
    orders.push(id)
    return `You have successfully ordered ${order} (${id})`
  })
}

function CancelOrderCommand(id) {
  return new Command(orders => {
    orders = orders.filter(order => order.id !== id)
    return `You have canceled your order ${id}`
  })
}

function TrackOrderCommand(id) {
  return new Command(() => `Your order ${id} will arrive in 20 minutes.`)
}

이로써 OrderManager가 메서드를 직접 갖는 대신 execute메서드를 통해 분리된 함수를 사용하도록 코드를 리펙토링 하였습니다.

 

장점

커멘드 패턴은 객체와 메서드를 분리할 수 있게 해 줍니다.

이렇게 분리하면 수명이 지정된 명령을 만들거나, 명령들을 큐에 담아 특정한 시간대에 처리하는 것도 가능해집니다.


단점

커멘드 패턴을 쓸만한 상황이 딱히 많지 않고 종종 불필요한 코드가 만들어지곤 한다.

 

참조

 
728x90
반응형