객체 확산 연산자 사용하기

Redux 의 핵심 규칙 중 하나가 절대 state를 직접 변경하지 않는 것이기 때문에 새로운 값이 추가되거나 업데이트 된 객체를 복사하기 위해서 Object.assign()을 자주 사용하게 될 것입니다. 예를 들면, todoApp 에서 아래의 Object.assign()visibilityFilter 프로퍼티가 업데이트 된 새로운 state 객체를 리턴하기 위해 쓰였습니다.

function todoApp(state = initialState, action) {
  switch (action.type) {
    case SET_VISIBILITY_FILTER:
      return Object.assign({}, state, {
        visibilityFilter: action.filter
      })
    default:
      return state
  }
}

효과적이긴 하지만, Object.assign() 의 장황한 문법 때문에 간단한 reducer를 읽기 어렵게 만듭니다.

또 다른 접근법은 다음 버전에 제안된 객체 확산 문법 object spread syntax를 사용하는 것입니다. 이것은 확산 연산자 (...)를 사용해서 객체에서 다른 객체로 열거 가능한 프로퍼티들을 좀 더 간결하게 복사 할 수 있습니다. 객체 확산 연산자 (Object Spread Operator)는 개념적으로 ES6의 배열 확산 연산자 array spread operator 와 비슷합니다. 우리는 위의 todoApp예제를 다음과 같이 객체 확산 문법을 사용하여 단순화 할 수 있습니다:

function todoApp(state = initialState, action) {
  switch (action.type) {
    case SET_VISIBILITY_FILTER:
      return { ...state, visibilityFilter: action.filter }
    default:
      return state
  }
}

객체 확산 문법의 장점은 복잡한 객체를 구성할 때 명확해 집니다.아래에서 getAddedIdsid값의 배열을 getProductgetQuantity에서 리턴된 값을 가진 객체의 배열에 맵핑합니다.

return getAddedIds(state.cart).map(id =>
  Object.assign({}, getProduct(state.products, id), {
    quantity: getQuantity(state.cart, id)
  })
)

객체 확산은 위의 map 호출을 다음과 같이 단순화 시켜줍니다:

return getAddedIds(state.cart).map(id => ({
  ...getProduct(state.products, id),
  quantity: getQuantity(state.cart, id)
}))

객체 확산 문법은 여전히 ECMAScript의 2단계 제안이므로 그것을 이용하려면 Babel 같은 트렌스파일러를 사용하셔야 됩니다. 기존의 es2015 프리셋을 이용 하실 수 있습니다. babel-plugin-transform-object-rest-spread를 설치하고 당신의 .babelrcplugins 배열에 개별적으로 추가하면 됩니다.

{
  "presets": ["es2015"],
  "plugins": ["transform-object-rest-spread"]
}

아직 실험적인 언어 기능 제안이기 때문에 미래에 바뀔 수 있다는 것을 알아두세요. 그럼에도 불구하고 React Native와 같은 일부 대규모 프로젝트에서는 이미 이 객체 확산 문법을 광범위하게 사용하기 때문에 혹시 변경 되더라도 좋은 자동 마이그레이션(migration) 경로가 있을 것 이라 말할 수 있습니다.

Last updated