상태 초기화하기

애플리케이션의 상태를 초기화하는 방법은 크게 두 가지가 있습니다. createStore 메소드는 추가적인 두 번째 인자로 preloadedState값을 받을 수 있습니다. 또한 리듀서는 undefined인 인자를 찾고 초깃값으로 사용할 값을 반환해서 초깃값을 지정할 수 있습니다. 이는 리듀서 내부에서 명시적으로 확인하거나 function myReducer(state = someDefaultValue, action) 처럼 ES6의 기본 인자 구문을 통해 사용할 수 있습니다.

이 두 가지 접근이 항상 명확한 것은 아닙니다. 다행히 프로세스는 예측 가능한 규칙을 따르고 있습니다. 여기서는 그 조각들을 어떻게 맞추는지 알아봅니다.

요약

리듀서에서 combineReducers()와 같은 코드가 없다면 preloadedState는 항상 state = ...보다 우선합니다. 리듀서에 전달된 state preloadedState이고 undefined아니기 때문입니다. 따라서 ES6 인수 구문이 적용되지 않습니다.

combineReducers()의 동작은 더욱 미묘합니다. preloadedState로 상태가 지정된 리듀서는 해당 상태를 받습니다. 다른 리듀서는 기본으로 정의한 state = ...로 돌아갈 것이기 때문에 undefined를 받습니다.

일반적으로 preloadedState는 리듀서로 부터 정의된 상태보다 우선시됩니다. 이는 리듀서에 초깃값으로 사용될 값을 지정할 수 있을 뿐만아니라 영구적인 저장소나 서버로 부터 스토어를 가져올 때 (전체 혹은 부분적으로) 이미 존재하는 데이터를 불러올 수 있습니다.

주의: preloadedState를 통해 초기상태가 지정된 리듀서또한 undefined인 상태를 처리하기위해 기본값이 필요합니다. 모든 리듀서에는 초기에 undefined가 전달됩니다. 따라서 반드시 undefined인 경우에 어떤 값이 반환될지가 작성되어야 합니다. 이는 undefined가 아닌 어떤 값이라도 가능합니다; 여기에 기본값으로 preloadedState의 섹션을 복사할 필요가 없습니다.

심화

간단한 단일 리듀서

먼저 리듀서가 하나인 경우를 생각해봅시다. combineReducers()를 사용하지않습니다.

이때 리듀서는 다음처럼 생겼을 겁니다.

function counter(state = 0, action) {
  switch (action.type) {
  case 'INCREMENT': return state + 1;
  case 'DECREMENT': return state - 1;
  default: return state;
  }
}

이제 스토어를 다음처럼 만들었다고 가정해봅시다.

import { createStore } from 'redux';
let store = createStore(counter);
console.log(store.getState()); // 0

초기 상태는 0입니다. 왜 그럴까요? createStore의 두 번째 인자가 undefined이기 때문입니다. 이는 리듀서에 처음으로 전달된 state입니다. 리덕스가 초기화될 때 디스패처는 상태를 채우기 위해 "더미" 액션을 전달합니다. 따라서 당신의 counter리듀서는 stateundefined인 상태로 호출되었습니다. 이는 기본 인자를 "활성화"하는 경우입니다. 따라서 state는 기본값 (state = 0)에 따라 0이 됩니다. 그리고 이 상태가 반환됩니다.

다른 시나리오를 생각해봅시다.

import { createStore } from 'redux';
let store = createStore(counter, 42);
console.log(store.getState()); // 42

이 경우에 왜 0이 아닌 42일까요? createStore는 두 번째 인자로 42와 함께 호출되었기 때문입니다. 이 인자는 당신의 리듀서에 더미 액션과 함께 전달되는 state가 됩니다. 이번에는 state가 undefined가 아닙니다.(42가 됩니다!) 그러므로 ES6의 기본 인자 문법은 효력이 없습니다. state42이고 42는 리듀서로부터 반환됩니다.

리듀서 결합

combineReducers()를 사용하는 경우를 생각해봅시다. 두 개의 리듀서가 있습니다:

function a(state = 'lol', action) {
  return state;
}

function b(state = 'wat', action) {
  return state;
}

combineReducers({ a, b })에 의해 만들어진 리듀서는 다음과같습니다:

// const combined = combineReducers({ a, b })
function combined(state = {}, action) {
  return {
    a: a(state.a, action),
    b: b(state.b, action)
  };
}

preloadedState없이 createStore를 부른다면 state{}로 초기화할겁니다. 따라서 ab리듀서가 불리는 시점에 state.a, state.bundefined로 초기화될겁니다. ab리듀서 둘 다 undefined그것들의 상태 인자로 받고, 기본 state값을 지정한다면 해당값이 반환될겁니다. 이것이 결합된 리듀서가 첫 호출에서 { a: 'lol', b: 'wat' } 상태 객체를 반환하는 방법입니다.

import { createStore } from 'redux';
let store = createStore(combined);
console.log(store.getState()); // { a: 'lol', b: 'wat' }

다른 시나리오를 생각해봅시다.

import { createStore } from 'redux';
let store = createStore(combined, { a: 'horse' });
console.log(store.getState()); // { a: 'horse', b: 'wat' }

preloadedStatecreateStore()의 인자로 지정했습니다. 결합된 리듀서에서 반환되는 상태는 a리듀서에 지정된 초기 상태를 b리듀서의 기본 인자로 선택된 'wat'과 결합합니다.**

결합된 리듀서가 하는 일을 다시 한번 생각해봅시다.

// const combined = combineReducers({ a, b })
function combined(state = {}, action) {
  return {
    a: a(state.a, action),
    b: b(state.b, action)
  };
}

이 경우에, state가 지정되었기 때문에 {}로 되돌아가지 않았습니다. a의 값은 horse이지만 b의 값이 없습니다. a리듀서가 state'horse'를 받아 반환했지만 b리듀서가 stateundefined를 받기 때문에 이를 기본 state로 반환합니다. 이것이 반환 값으로 { a: 'horse', b: 'wat' }를 얻을 수 있는 이유입니다.

요약

요약하면, 리덕스의 컨벤션을 지키고 undefinedstate인자로 호출했을 때 리듀서의 초기 상태를 반환하려면 (이를 구현하기 위한 가장 쉬운 방법은 state를 ES6의 기본 인자 값을 이용하는것) 결합된 리듀서를 위해 적절한 일을 해야 합니다. createStore()함수에 전달된 preloadedState객체에서 상응하는 값을 넘겨주겠지만, 아무것도 전달하지 않거나 대응되는 값이 없다면 기본 state 인자가 초깃값으로 대신 지정됩니다. 이 방법은 기존 데이터를 주입, 초기화하고 데이터가 보존되지 않는 경우 각 상태를 다시 설정할 수 있다는 점에서 효과적입니다. 물론 combineReducers()를 많은 단계에서 사용해서 이 패턴을 재귀적으로 사용할 수도 있습니다. 혹은 관련 상태 트리의 일부를 전달함으로써 부분적으로 사용할 수도 있습니다.

Last updated