대표적인 상태관리 라이브러리인 Redux
, Mobx
, Recoil
에 대해 알아보자
개발을 시작할 때만 하더라도 상태관리 라이브러리는 Redux 하나라고 생각했는데, 꽤나 많은 것들이 생겨났다. (이미 있었지만 몰랐을 수도..)
하지만 여전히 Redux를 가장 많이 사용하고 있다고 생각한다. 그 다음은 Mobx라고 할 수 있겠다. 먼저 생겨난 순으로 사용률이 높은 것은 당연한 이치인 것처럼.
최근에 페이스북이 만들어낸 상태관리를 위한 Recoil이 뜨고 있다. 나는 Redux만 사용해봤기 때문에 각각의 특징에 대해 보며 어떠한 부분으로 Recoil이 등장하였고 주목받고 있는지에 대해 초점을 맞춰 알아보려고 한다.
npm trends
를 보면 역시나 react-redux가 사용률이 압도적으로 높은 것을 알 수 있다. 하지만 mobx와 recoil 모두 꾸준한 증가세를 보이고 있는 것을 알 수 있다.
먼저 Redux에 대해 살펴보자면 아래 다이어그램을 통해 흐름을 이해할 수 있다. Redux를 사용하기 위해서는 Action, Reducer, Dispatcher, Store, View 등에 대한 개념을 이해해야한다. 위의 개념들을 통하여 여러 컴포넌트에서 사용되는 state를 분리 통합하여 관리할 수 있게 하고 애플리케이션의 안정성을 높일 수 있는 라이브러리이다.
리덕스 라이브러리는 단일 스토어가 있는 것이 권장되며 읽기 전용 상태이다. 리덕스도 상태 업데이트 시 기존 객체를 건드리지 않고 새로운 객체를 생성해야 한다. 리덕스에서 불변성을 유지해야 하는 이유는 내부적으로 데이터 변경을 감지하기 위함이다.
리덕스에서 Action은 state를 바꾸는 방식이다. 액션 객체를 가지고 있고, 반드시 type필드가 있어야한다.
Action을 발생시키는 것으로 action 객체를 파라미터로 받는다.
변화를 일으키는 함수로 Action의 결과로 state를 어떤 식으로 바꿀지 구체적으로 정의하는 부분이다. 리듀서가 현재 상태와 전달받은 액션 객체를 파라미터로 받아서 새로운 상태로 반환해준다. 리듀서는 파라미터 외의 값을 의존하면 안되고, 이전 상태는 건드리지 않은 상태로 새로운 상태 객체를 만들어 반환하는 순수 함수여야 한다.
프로젝트에 리덕스를 적용하기 위해 필요한 것으로 프로젝트에는 단 한 개의 Store만 가지며 상태의 중앙 저장소라고 할 수 있다. Store안에는 리듀서와 내장 함수 등이 포함되어 있다. dispatch
와 subscribe
등도 스토어의 내장 함수이다. 상태를 읽을 때는 getState()
, 상태를 바꿀 때는 dispatch()
를 호출한다.
Mobx는 사실 아직 써보지는 않았다. Redux와 비슷한 상태관리 라이브러리이지만, Redux에 비해 간결하고 깔끔한 구조를 가지고 있다는 평가를 받는다고 한다. Mobx의 컨셉은 아래와 같다. 가장 큰 특징은 모든 상태 변화가 일어나는 부분을 자동으로 추적해주는 것이다.
redux의 dispatch, action의 개념에 익숙한 유저라면 약간 생소할 수 있다. 각각의 상태를 좀 더 자세히 살펴보자.
어플리케이션의 데이터 상태이다. 개념적으로 mobx는 spreadsheet와 유사하다고 한다. objects, arrays, primitives, references의 그래프의 형태로 어플리케이션을 구성하게 된다. 이것들은 spreadsheet의 data cells이 될 것이다. Observable State로 관찰 받고 있는 데이터이다. Mobx에서는 해당 State가 관찰하고 있다가 변화가 일어나면 Reactions와 Derivations를 발생시킨다.
기존의 상태가 변화에 따라 계산된(파생된) 값을 의미한다.
Observable State의 변화에 따른 부가적인 변화를 의미하고 값이 바뀜에 따라 해야할 일을 정하는 것이다. Derivation과는 달리 값을 생성하지는 않고, 대체로 I/O와 관련된 작업을 하고, DOM 업데이트와 네트워크 요청 등에 관여한다.
상태를 변경시키는 모든 것을 의미한다. Mobx에서는 모든 사용자의 액션으로 발생하는 상태 변화들이 전부 Derivations와 Reactions로 처리되도록 한다.
mapStateToProps
,mapDispatchToProps
등의 보일러 플레이트 코드가 사라지고 데코레이터를 통해 깔끔한 코드 작성이 가능하다. Mobx 스토어와 React 컴포넌트를 연결하는게 @inject 데코레이터 하나로 가능하다.많은 React 상태 관리 라이브러리들이 있고, 가끔 새로운 라이브러리가 등장한다. 그러나 페이스북에서 직접 상태 관리 솔루션을 소개하는 것은 흔하지 않다.
Recoil은 Context API기반으로 구현된 함수형 컴포넌트에서만 사용 가능한 페이스북에서 만든 라이브러리이다. 호환성이나 단순함을 위해선 React에 내장된 상태 관리 기능을 사용하는 것이 가장 바람직하다고 할 수 있다. 예를 들을 Hooks나 Context API를 사용하여 상태 관리를 할 수 있는데, 그런 경우 여러가지 한계가 존재한다.
이러한 상황에서 Recoil은 React스러움을 유지하며 개선하는 방식의 라이브러리이다. Recoil은 방향 그래프를 정의하고 리액트 트리를 붙이는데, 이 그래프의 뿌리(atom)으로 부터 순수 함수(selector)를 거쳐 컴포넌트로 흐른다.
Atoms는 Recoil에서 상태의 단위를 의미하고, 업데이트와 구독이 가능하다. atom이 업데이트되면 각각의 구독된 컴포넌트는 새로운 값을 반영하여 리렌더링 된다.
Atoms는 리액트의 로컬 state 대신 사용할 수 있다. 동일한 atom이 여러 컴포넌트에서 사용되는 경우 모든 컴포넌트 상태를 공유한다. Atoms에는 코유한 키가 필요하고 이 키는 전역적으로 고유해야한다. 그리고 react state처럼 디폴트 값도 가진다.
const exampleState = atom({
key: 'exampleState',
default: null,
)}
컴포넌트에서 atom을 읽고 쓸 때는 useRecoilState
라는 훅을 사용해야한다. 이건 리액트의 useState
와 비슷하나, 상태가 컴포넌트간에 공유될 수 있다는 점에서 차이가 있다. 아래 두 컴포넌트의 state는 공유된다.
function ExampleButton() {
const [exampleSize, setExampleSize] = useRecoilState(exampleState)
return (
<button
onClick={() => setExampleSize(size => size + 1)}
style={{ exampleSize }}
>
Click to Enlarge
</button>
)
}
function Text() {
const [exampleSize, setExampleSize] = useRecoilState(exampleState)
return <p style={{ exampleSize }}>This text will increase in size too.</p>
}
Selector는 atoms나 다른 selectors를 입력으로 받는 순수 함수(pure function)이다. 상위 atoms이나 selectors가 업데이트 될 경우 하위 selectors도 재실행된다. 컴포넌트는 atom 뿐만 아니라 selectors를 구독할 수 있고, 구독하고 있는 selectors가 변경되면 구독한 컴포넌트도 리렌더링된다. Selectors는 상태를 기반으로 데이터를 계산하고 최소한의 상태 집합만 atoms에 저장하고, 파생 데이터는 selector에서 계산하면서 불필요한 상태를 만들어내지 않는다. 컴포넌트 관점에서 atoms와 selectors는 동일한 인터페이스이므로 대체 가능하다.
const exampleLabelState = selector({
key: 'exampleState',
get: ({ get }) => {
const exampleSize = get(exampleState)
const unit = 'px'
return `${exampleSize}${unit}`
},
})
여기서 get
속성은 계산될 함수를 의미하고 전달되는 get
인자를 통해 atoms와 다른 selectors에 접근 가능하다. 여기서 접근하면 자동으로 종속 관계가 생성되어 참조했던 atoms나 selectors가 업데이트되면 해당 함수도 재실행된다.
Selectors는 useRecoilValue()
를 통해 조회 가능하다. useRecoilState
와는 다르게 writable하지 않고, 반환 값의 조회만 가능하다. 필요하다면 writable한 selector 작성도 가능하다.
Concurrent Mode: 흐름이 여러 개가 존재하는 경우이다. 리액트에서 렌더링의 동작 우선순위를 정하여 적절한 때에 렌더링을 해준다.
기존 Redux가 주로 사용되면서 불편함들을 개선하기 위해, 또는 다른 방식으로 문제를 해결하기 위해 Mobx가 등장하였고, 또 뒤의 두 라이브러리의 불편함이나 문제를 개선하기 위해 Recoil이 등장했다고 생각한다. 각각의 라이브러리의 장점과 단점이 있지만, Recoil은 기존 React 내장 Hooks의 사용하는 방식과 크게 다르지 않고 유사하기 때문에 쉽게 배우고 활용할 수 있다는 부분에서 큰 장점이 있다고 생각한다. 그리고 그러면서도 전역 상태관리가 더 쉽고 편리하게 이루어질 수 있다는 점에서 앞으로 점점 더 많이 사용될 것으로 예상해본다.