9. 여러개의 input 상태 관리하기
- TOC {:toc}
이 글은 벨로퍼트와 함께하는 모던 React중 9. 여러개의 input 상태 관리하기의 내용을 복습하기위해 핵심 내용을 요약 정리한 글입니다.
- 제가 필요한 부분 위주로 확인하면서 정리하고 있어 글에 덜 작성된 부분이 있을 수 있습니다.
- 글 작성 후 원문의 내용이 수정되거나 내용을 이해하기 위한 개인적인 설명이나 해석이 있을 수 있습니다. 되도록 원문을 참고해주시길 바랍니다.
- 잘못된 부분이 있다면 댓글이나 그 외 편하신 방법으로 알려주시면 감사하겠습니다.
input 의 개수가 여러개가 됐을때는, input 에 name 을 설정하고 이벤트가 발생했을 때 이 값을 참조하는 것이 좋다.
더불어, useState
에서는 문자열이 아니라 객체 형태로 state를 관리해주어야 한다.
/* InputSample.js */
function InputSample() {
// inputs의 초기값: {name: '', nickname: ''}
// inputs.name과 inputs.nickname 모두 공백
const [inputs, setInputs] = useState({
name: '',
nickname: ''
});
// 비구조화 할당을 통해 값 추출
const { name, nickname } = inputs;
const onChange = (e) => {
// 우선 e.target 에서 name 과 value 를 추출
// 이 때 name은 위에서 추출한 name이 아니라 input의 name property의 값이다.
// 아래의 name과 헷갈리는 것을 막기 위해 input_name으로 수정.
const { value, input_name } = e.target;
setInputs({
// 기존의 input 객체를 복사한 뒤
...inputs,
// inputs 객체에서 input_name 키를 가진 값을 value 로 설정
// e.g., 이벤트가 첫 번째 input에서 발생했으면 input_name이 name이기 때문에
// inputs중 name의 값을 value로 변경.
// name이 이미 inputs 안에 존재하는 값이라 새로운 key: value 값이 추가되는 것이 아니라 기존의 값을 변경
// 변경된 값은 rendering 되면서 바로 반영된다.
[input_name]: value
});
};
const onReset = () => {
// inputs를 초기화: name과 nickname 값을 다시 초기값 (blank)로 설정
setInputs({
name: '',
nickname: '',
})
};
return (
<div>
<input name="name" placeholder="이름" onChange={onChange} value={name} />
<input name="nickname" placeholder="닉네임" onChange={onChange} value={nickname}/>
<button onClick={onReset}>초기화</button>
<div>
<b>값: </b>
{name} ({nickname})
</div>
</div>
);
}
- React 에서 객체를 수정할 때는
inputs[name] = value
처럼 직접 수정하면 안된다. - 대신에, 새로운 객체를 만들어서 새로운 객체를 수정한 뒤, 이를 state로 사용해주어야 한다.
- 만약에 기존 상태를 직접 수정하게 되면, 값을 바꿔도 리렌더링이 되지 않는다.
참고 (-ing)
자바스크립트라는 언어의 메모리 구조 때문이라고 생각됩니다. https://junwoo45.github.io/2019-11-04-memory_model/ 배열같은경우, push 를 해도 해당 객체의 메모리 주소는 변하지 않습니다. 그래서 리액트가 객체의 변화를 식하지 못한다고 생각되네요. 그래서 기존 객체를 직접 수정하지 않고, 복사본을 만들어 새로운 객체를 할당하는게 아닐까요?
어떤 프로그래밍 언어 / 런타임이든지 “변경” 감지에 대한 패턴은 여러가지가 있습니다. 단순히 자바스크립트라서 불가능한 것 / 메모리 주소가 바뀌지 않아서 불가능한 것은 아닙니다. 리액트는 좀 더 단순하고 직관적인 방향으로 변경감지를 하고 싶었던 것 같으며, 다른 angular 혹은 vue와 다르게 setState 함수 혹은 useState의 setValue 인자 함수를 이용하여 변경감지를 합니다. 해당 함수가 호출되는 즉시 상태에 대한 변경이 일어낫다고 판단하는 것 같습니다. 번외로 vue나 angular는 내부적으로 프로퍼티에 getter setter를 사용하여 runtime proxy를 생성하여 해당 프로퍼티에 setter가 호출되면 변경이 되었다고 인식하는 형태로 되어있습니다.
-
draft