리액트 - state hook

아침7시개발·2022년 1월 21일
0

리액트

목록 보기
12/16

Using the State Hook

import React, { useState } from 'react';

function Example() {
  // 새로운 state 변수(count)를 선언
  const [count, setCount] = useState(0);

  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>
        Click me
      </button>
    </div>
  );
}

Hook과 같은 기능을 하는 클래스 예시

class Example extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 0
    };
  }

  render() {
    return (
      <div>
        <p>You clicked {this.state.count} times</p>
        <button onClick={() => this.setState({ count: this.state.count + 1 })}>
          Click me
        </button>
      </div>
    );
  }
}

위 코드에서 state는 { count: 0 }이며 사용자가 this.setState()를 호출하는 버튼을 클릭했을 때 state.count를 증가시킨다.


Hook과 함수 컴포넌트

React의 함수 컴포넌트는 이렇게 생겼다.

const Example = (props) => {
  // 여기서 Hook을 사용할 수 있다!
  return <div />;
}

또는 이렇게 생겼다.

function Example(props) {
  // 여기서 Hook을 사용할 수 있다!
  return <div />;
}

함수 컴포넌트를 state가 없는 컴포넌트로 알고 있었을 것이다. 하지만 Hook은 React state를 함수 안에서 사용할 수 있게 한다.

Hook은 클래스 안에서 동작하지 않는다. 하지만 클래스를 작성하지 않고 사용할 수 있다.


Hook이란?

React의 useState Hook을 사용해보자!

import React, { useState } from 'react';

function Example() {
  // ...
}

Hook이란? Hook은 특별한 함수다. 예를 들어 useState는 state를 함수 컴포넌트 안에서 사용할 수 있게 한다.

언제 Hook을 사용할까? 함수 컴포넌트를 사용하던 중 state를 추가하고 싶을 때 클래스 컴포넌트로 바꾸곤 했을 것이다. 하지만 이제 함수 컴포넌트 안에서 Hook을 이용하여 state를 사용할 수 있다.


state 변수 선언하기

클래스를 사용할 때, constructor 안에서 this.state{ count: 0 }로 설정함으로써 count0으로 초기화했다.

class Example extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 0
    };
  }

함수 컴포넌트는 this를 가질 수 없기 때문에 this.state를 할당하거나 읽을 수 없다. 대신, useState Hook을 직접 컴포넌트에 호출한다.

import React, { useState } from 'react';

function Example() {
// 새로운 state 변수를 선언하고, 이것을 count라 부른다.
const [count, setCount] = useState(0);

useState를 호출하는 것은 무엇을 하는 걸까? state 변수를 선언할 수 있다. 위에 선언한 변수는 count라고 부르지만 banana처럼 아무 이름으로 지어도 된다. useState는 클래스 컴포넌트의 this.state가 제공하는 기능과 똑같다. 일반적으로 일반 변수는 함수가 끝날 때 사라지지만, state 변수는 React에 의해 사라지지 않는다.

useState의 인자로 무엇을 넘겨주어야 할까? useState()Hook의 인자로 넘겨주는 값은 state의 초기 값이다. 함수 컴포넌트의 state는 클래스와 달리 객체일 필요는 없고, 숫자 타입과 문자 타입을 가질 수 있다. 위의 예시는 사용자가 버튼을 얼마나 많이 클릭했는지 알기를 원하므로 0을 해당 state의 초기 값으로 선언했다. (2개의 다른 변수를 저장하기를 원한다면 useState()를 두 번 호출해야 한다.)

useState는 무엇을 반환할까? state 변수, 해당 변수를 갱신할 수 있는 함수 이 두 가지 쌍을 반환한다. 이것이 바로 const [count, setCount] = useState()라고 쓰는 이유다. 클래스 컴포넌트의 this.state.countthis.setState와 유사하다.


state 가져오기

클래스 컴포넌트는 count를 보여주기 위해 this.state.count를 사용한다.

  <button onClick={() => this.setState({ count: this.state.count + 1 })}>
  Click me
</button>

반면 함수 컴포넌트는 setCount와 count 변수를 가지고 있으므로 this를 호출하지 않아도 된다.

  <button onClick={() => setCount(count + 1)}>
  Click me
</button>

요약

1:  import React, { useState } from 'react';
2:
3:  function Example() {
4:    const [count, setCount] = useState(0);
5:
6:    return (
7:      <div>
8:        <p>You clicked {count} times</p>
9:        <button onClick={() => setCount(count + 1)}>
10:         Click me
11:        </button>
12:      </div>
13:    );
14:  }
  • 첫 번째 줄: useState Hook을 React에서 가져온다.
  • 네 번째 줄: useState Hook을 이용하면 state 변수와 해당 state를 갱신할 수 있는 함수가 만들어진다. 또한, useState의 인자의 값으로 0을 넘겨주면 count 값을 0으로 초기화할 수 있다.
  • 아홉 번째 줄: 사용자가 버튼 클릭을 하면 setCount 함수를 호출하여 state 변수를 갱신한다. React는 새로운 count 변수를 Example 컴포넌트에 넘기며 해당 컴포넌트를 리렌더링한다.

팁: 대괄호가 의미하는 것은 무엇일까?

대괄호 왼쪽의 state 변수는 사용하고 싶은 이름으로 선언할 수 있다.

 const [fruit, setFruit] = useState('banana');

위 자바스크립트 문법은 배열 구조 분해라고 하고, fruit과 setFruit, 총 2개의 값을 만들고 있다. 즉, useState를 사용하면 fruit이라는 첫 번째 값과 setFruit라는 두 번째 값을 반환한다. 아래의 코드와 같은 효과를 낼 수 있다.

 var fruitStateVariable = useState('banana'); // 두 개의 아이템이 있는 쌍을 반환
  var fruit = fruitStateVariable[0]; // 첫 번째 아이템
  var setFruit = fruitStateVariable[1]; // 두 번째 아이템

팁: 여러 개의 state 변수를 사용하기

[something, setSomething]의 쌍처럼 state 변수를 선언하는 것은 유용하다. 왜냐하면 여러 개의 변수를 선언할 때 각각 다른 이름을 줄 수 있기 때문이다.

function ExampleWithManyStates() {
// 여러 개의 state를 선언할 수 있다!
const [age, setAge] = useState(42);
const [fruit, setFruit] = useState('banana');
const [todos, setTodos] = useState([{ text: 'Learn Hooks' }]);

의 코드는 age, fruit, todos라는 지역 변수를 가지며 개별적으로 갱신할 수 있다.

  function handleOrangeClick() {
  // this.setState({ fruit: 'orange' })와 같은 효과를 낸다.
  setFruit('orange');
}

여러 개의 state 변수를 사용하지 않아도 된다. state 변수는 객체와 배열을 잘 가지고 있을 수 있으므로 서로 연관있는 데이터를 묶을 수 있다. 하지만 클래스 컴포넌트의 this.setState와 달리 state를 갱신하는 것은 병합하는 것이 아니라 대체하는 것이다.

profile
쉬엄쉬엄하는 개발자

0개의 댓글