inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

치킨값으로 배우는 JS 고급

useState 직접 구현

useState 직접 구현 부분에서 질문이 있습니다.

8

taluee

작성한 질문수 2

1

setCount가 전역변수에 없으면 setCount is not defined; 오류가 발생합니다.
Codex가 Counter()는 문자열을 반환하고 setCount는 Counter의 지역변수라서 브라우저가 setCount를 전역에서 찾기 때문에 나타나는 오류라고 합니다. <--이부분이 어렵습니다.
아래처럼 변경하거나 addEventListener의 콜백으로 구현하라고 하네요.

const states = [];
    let cursor = 0;
    let setCount;

    function useState(initialValue) {
      const index = cursor;
      cursor++;

      if (states[index] === undefined) {
        states[index] = initialValue;
      }

      const setState = (newValue) => {
        states[index] = newValue;
        cursor = 0;
        render();
      };

      return [states[index], setState];
    }

    function Counter() {
      const [count, _setCount] = useState(0);
      setCount = _setCount;

      return `<button onClick="setCount(${count + 1})">${count}</button>`;
    }

    function render() {
      const root = document.querySelector("#root");
      cursor = 0;
      root.innerHTML = Counter();
    }

    render();

 

javascript 웹앱 함수형-프로그래밍 es6 객체지향 react.js vanilla-javascript

답변 1

0

포테토쌤

안녕하세요! 질문 주셔서 감사합니다 :)

헷갈릴 수 있는 지점을 짚어주셔서 감사합니다.

먼저 해당 예제는 useState의 내부 아이디어를 가장 단순하게 보여주기 위한 개념 설명용 수도코드에 가까운 예제였습니다.
즉, “상태를 외부에 저장하고, setter가 상태를 바꾼 뒤 다시 render를 호출한다”는 흐름을 보여주는 데 초점을 둔 코드입니다.

다만 말씀해주신 것처럼, 이 코드를 브라우저에서 그대로 실행하면 setCount is not defined 오류가 발생할 수 있습니다.

문제는 이 부분입니다.


return `<button onclick="setCount(${count + 1})">${count}</button>`;

onclick="setCount(...)"는 HTML 문자열 안에 들어간 인라인 이벤트 핸들러입니다.

innerHTML로 DOM에 들어간 뒤 버튼을 클릭하면, 브라우저가 이 문자열을 자바스크립트 코드처럼 실행합니다.

이때 setCountCounter 함수 내부의 지역 변수로 찾는 것이 아니라, 전역(window) 스코프에서 찾게 됩니다.

그런데 setCount는 아래처럼 Counter 함수 안에서 만들어진 지역 변수입니다.

function Counter() {
  const [count, setCount] = useState(0);
}

그래서 클릭 시점에 브라우저가 window.setCount를 찾다 못 찾아 setCount is not defined 오류가 발생합니다.

해결 방법은 말씀 주신 것처럼 크게 두 가지입니다.

  1. setCount를 바깥 스코프에 선언해서 전역에서 접근 가능하게 만들기

  2. addEventListener로 이벤트 핸들러를 직접 연결하기



인라인 문자열 핸들러의 전역 스코프 문제를 피할 수 있고, 이벤트 콜백이 setCount를 클로저로 기억하기 때문에 addEventListener방식을 이용하는게 더 자연스러워 보입니다. ㅎㅎ

예를 들면 다음처럼 바꿀 수 있습니다.

// 전역 변수로 상태 저장 시도
let _state;

function useState(initialValue) {
  if (_state === undefined) {
    _state = initialValue;
  }

  const setState = (newValue) => {
    _state = newValue;
    render();
  };

  return [_state, setState];
}

function Counter() {
  const [count, setCount] = useState(0);

  const root = document.querySelector("#root");

  root.innerHTML = `<button id="btn">${count}</button>`;

  root.querySelector("#btn").addEventListener("click", () => {
    setCount(count + 1);
  });
}

function render() {
  Counter();
}

render();

개념 설명을 위해 단순화하다 보니, 실행 환경의 스코프 문제를 고려하지 못했네요 ㅠㅠ

강의 자료도 addEventListener 방식으로 보완하여 업로드하였습니다.(다시 다운로드 부탁드립니다!)

좋은 질문 주셔서 감사합니다. :)

학습을 하고 블로그에 정리를 해도 괜찮을까요?

1

12

1

IO활용-회원관리예제1 샘플코드 문의

0

6

1

교재(3쇄)와 강의 내용 문의

0

21

2

node js 설치 시 npm이 계속 안됩니다.

0

21

1

중급 1편 학습 방향에 대해 조언 부탁드립니다

0

28

1

SingletonService가 JVM이 뜰 때 생성되는게 맞나요?

0

29

1

섹션2번 부분 강의 화면이 잘 못된것 같아서 문의합니다.

0

23

1

call stack 표현이 잘못표현된것이 아닌가요?

0

55

2

7.5 강의에서 settings.json 붙여 넣기 내용이 영상과 다릅니다

0

53

2

React 와 Virtual DOM 의 이야기 영상 실행이 안됩니다.

0

34

1

상태(State) 가 "시간이 지남~" 에 대해 질문 있습니다.

0

32

2

PPT 자료 공유 받을 수 있을까요?

0

31

1

본 강의와는 상관없는 내용입니다만..

0

35

1

왜 클로드.md 파일에 프롬프트를 넣는건지 궁금합니다

0

55

1

강의 일정 및 수업 자료 공유 관련 문의드립니다!

0

34

1

import {} 중괄호 차이점

0

38

2

일반 강의와 차이점?

1

52

1

!= 연산자의 역할

0

35

1

[ 문의]몽고DB connect 의 건

0

47

2

AI 권한 부여가 안됩니다.

0

60

2

중요하진 않지만 설명하신부분에서 안된부분..

1

34

1

제공되는 react_code.zip 중에 ..

0

47

2

강의 듣는 순서가 어떻게 되나요?

0

48

1

김영한로드맵으로 백엔드개발자가 될 수 있나요?

0

105

2