useState 직접 구현 부분에서 질문이 있습니다.
8
작성한 질문수 2
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();
답변 1
0
안녕하세요! 질문 주셔서 감사합니다 :)
헷갈릴 수 있는 지점을 짚어주셔서 감사합니다.
먼저 해당 예제는 useState의 내부 아이디어를 가장 단순하게 보여주기 위한 개념 설명용 수도코드에 가까운 예제였습니다.
즉, “상태를 외부에 저장하고, setter가 상태를 바꾼 뒤 다시 render를 호출한다”는 흐름을 보여주는 데 초점을 둔 코드입니다.
다만 말씀해주신 것처럼, 이 코드를 브라우저에서 그대로 실행하면 setCount is not defined 오류가 발생할 수 있습니다.
문제는 이 부분입니다.
return `<button onclick="setCount(${count + 1})">${count}</button>`;onclick="setCount(...)"는 HTML 문자열 안에 들어간 인라인 이벤트 핸들러입니다.
innerHTML로 DOM에 들어간 뒤 버튼을 클릭하면, 브라우저가 이 문자열을 자바스크립트 코드처럼 실행합니다.
이때 setCount를 Counter 함수 내부의 지역 변수로 찾는 것이 아니라, 전역(window) 스코프에서 찾게 됩니다.
그런데 setCount는 아래처럼 Counter 함수 안에서 만들어진 지역 변수입니다.
function Counter() {
const [count, setCount] = useState(0);
}그래서 클릭 시점에 브라우저가 window.setCount를 찾다 못 찾아 setCount is not defined 오류가 발생합니다.
해결 방법은 말씀 주신 것처럼 크게 두 가지입니다.
setCount를 바깥 스코프에 선언해서 전역에서 접근 가능하게 만들기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





