컴포넌트 State
setState
는 어떤 일을 하나요?
setState()
는 컴포넌트의 state
객체에 대한 업데이트를 실행합니다. state가 변경되면, 컴포넌트는 리렌더링됩니다.
state
와 props
의 차이점은 무엇인가요?
props
(“properties”의 줄임말) 와 state
는 일반 JavaScript 객체입니다. 두 객체 모두 렌더링 결과물에 영향을 주는 정보를 갖고 있는데, 한 가지 중요한 방식에서 차이가 있습니다. props
는 (함수 매개변수처럼) 컴포넌트에 전달되는 반면 state
는 (함수 내에 선언된 변수처럼) 컴포넌트 안에서 관리됩니다.
언제 props
와 state
를 사용하는지 더 알고 싶다면 아래의 자료를 확인해보세요.
왜 setState
가 잘못된 값을 주는 걸까요?
React에서 this.props
와 this.state
는 모두 렌더링된 값을 나타냅니다. 다시 말해 현재 화면에 보이는 것을 말합니다.
setState
호출은 비동기적으로 이뤄집니다. 따라서 setState
호출 직후 새로운 값이 this.state
에 반영될 거라고 믿어서는 안 됩니다. 이전 state 값을 기준으로 값을 계산해야 한다면 객체 대신 updater 함수를 전달하세요. (자세한 내용은 아래를 확인하세요.)
예시 코드는 예상대로 동작하지 않을 것입니다.
incrementCount() {
// 주의: 이 코드는 예상대로 동작하지 *않을 것*입니다.
this.setState({count: this.state.count + 1});
}
handleSomething() {
// `this.state.count`가 0에서 시작한다고 해봅시다.
this.incrementCount();
this.incrementCount();
this.incrementCount();
// React가 컴포넌트를 리렌더링할 때 `this.state.count`는 3이 될 것 같은 예상과 달리 1이 됩니다.
// 이것은 `incrementCount()` 함수가 `this.state.count`에서 값을 읽어 오는데
// React는 컴포넌트가 리렌더링될 때까지 `this.state.count`를 갱신하지 않기 때문입니다.
// 그러므로 `incrementCount()`는 매번 `this.state.count`의 값을 0으로 읽은 뒤에 이 값을 1로 설정합니다.
// 이 문제의 해결 방법은 아래에 설명되어 있습니다.
}
이 문제를 어떻게 해결하는지 알아봅시다.
어떻게 하면 이전 state 값을 기준으로 state 값을 업데이트할 수 있나요?
항상 setState
가 가장 최신의 state 값을 사용하도록 보장하기 위해서는 setState
에 객체 대신 함수를 전달하세요. (아래를 참조하세요.)
setState
에 객체를 전달하는 것과 함수를 전달하는 것은 어떤 차이가 있나요?
updater 함수를 전달하면 updater 함수 안에서 이전 state 값에 접근할 수 있습니다. setState
호출은 일괄적으로 처리되기 때문에 여러 업데이트 사항이 충돌 없이 차례대로 반영되도록 합니다.
incrementCount() {
this.setState((state) => {
// 중요: 값을 업데이트할 때 `this.state` 대신 `state` 값을 읽어옵니다.
return {count: state.count + 1}
});
}
handleSomething() {
// `this.state.count`가 0에서 시작한다고 해봅시다.
this.incrementCount();
this.incrementCount();
this.incrementCount();
// 지금 `this.state.count` 값을 읽어 보면 이 값은 여전히 0일 것입니다.
// 하지만 React가 컴포넌트를 리렌더링하게 되면 이 값은 3이 됩니다.
}
언제 setState
가 비동기적인가요?
현재 setState
는 이벤트 핸들러 내에서 비동기적입니다.
이로 인해 부모
와 자식
이 모두 click 이벤트에서 setState
를 호출한다면 자식
은 두 번 렌더링되지 않습니다. 대신 React는 브라우저 이벤트가 끝날 시점에 state를 일괄적으로 업데이트합니다. 이는 더 큰 규모의 앱에서 뚜렷한 성능 향상을 만들어냅니다.
이것은 구현 세부사항이므로 전적으로 의존해선 안됩니다. 추후 React는 기본적으로 더 많은 경우에서 일괄적으로 업데이트를 처리할 예정입니다.
왜 React는 this.state
를 동기적으로 업데이트하지 않나요?
이전 절에서 설명했듯이 모든 컴포넌트가 자신의 이벤트 핸들러에서 setState()
를 호출할 때까지 React는 리렌더링을 하지 않고 내부적으로 “기다리고 있습니다”. 이를 통해 불필요한 렌더링을 방지하면서 성능을 향상시킵니다.
그러나 왜 React는 리렌더링 대신 즉시 this.state
를 업데이트하지 않는지 여전히 궁금해 하실 수도 있습니다.
여기에는 두 가지 중요한 이유가 존재합니다.
props
와state
사이의 일관성을 해칠 수 있으며 이것은 디버깅하기 매우 힘든 이슈를 일으킬 수 있기 때문입니다.- 현재 작업 중인 새로운 기능들을 구현하기 힘들게 만들 수 있기 때문입니다.
이 GitHub 코멘트에서 더욱 자세한 예시를 확인할 수 있습니다.
Redux 나 MobX 같은 상태(‘state’) 관리 라이브러리를 사용해야 하나요?
추가적인 라이브러리를 사용하기 전에 먼저 React에 익숙해지는 게 좋습니다. React만으로도 꽤 복잡한 애플리케이션을 만들 수 있습니다.