useCallback, useMemo의 차이에 대해서 궁금합니다.
[셀렉트박스 (3/5) headless #2 hook 적용] 부분에서 getTriggerProps/getListProps 처럼 각 컴포넌트가 필요한 것들을 useCallback으로 감싸고 함수형태로 제공해주셨는데요, 아래처럼 useMemo를 이용해서 객체에 값을 담아서 전달해주는 방식은 다른 걸까요??
const getTriggerProps = useCallback(
() => ({
selectedItem: items[selectedIndex],
toggle,
}),
[selectedIndex, items, toggle]
);
const TriggerpropsValue = useMemo(() => {
return {
selectedItem: items[selectedIndex],
toggle,
};
}, [items, selectedIndex, toggle]);
回答 1
1
경우에 따라 다를 것 같습니다만, 제가 이해하기로는
headless 컴포넌트 적용시에 useMemo가 아닌 useCallback을 적용하는 이유는
컴포넌트간의 렌더링 간섭을 최소화하기 위함이 클 것 같습니다.
const useDropdown = () => {
...
return { getContainerProps, getTriggerProps, ... }
}
const Dropdown = () => {
const { getContainerProps, getTriggerProps } = useDropdown()
return (
<DDContainer {...getContainerProps()}>
<DDTrigger {...getTriggerProps()} />이렇게 각각의 컴포넌트에 넘겨줄 대상을 함수 호출로 해두면getContainerProps의 내용이 변경되더라도 getTriggerProps에는 영향을 주지 않을 것입니다.
반면 각각을 useMemo로 만들 경우에는,
const useDropdown = () => {
...
return { handleKeyDown, selectedItem, toggle, ... }
}
const Dropdown = () => {
const { handleKeyDown, selectedItem, toggle } = useDropdown()
return (
<DDContainer handleKeyDown={handleKeyDown}>
<DDTrigger selectedItem={selectedItem} toggle={toggle} />이렇게 되어서, handleKeyDown, selectedItem, toggle 중 어느 하나만 변경되더라도
모든 컴포넌트가 재렌더링될 것 같습니다.
혹은 다음과 같이 할 수도 있겠는데요,
const useDropdown = () => {
...
return { containerProps, triggerProps }
}
const Dropdown = () => {
const { containerProps, triggerProps } = useDropdown(data)
return (
<DDContainer {...containerProps}>
<DDTrigger {...triggerProps} />이 경우에는 테스트 해보니 괜찮네요.
다만 테스트하면서 보니, getItemProps의 경우에는 파라미터를 전달해야 해서 useCallback을 쓸 수밖에 없더군요.
결론은,
파라미터를 전달해야 하는 경우와 그렇지 않은 경우를 나누어
useCallback/useMemo를 구분 적용하는 것도 괜찮고,파라미터 필요성을 염두에 두어 일관되게
useCallback만 사용하는 방법도 좋을 것 같습니다.
0
상세한 설명 감사합니다. useCallback과 useMemo의 차이가 메모이제이션 할 값이 함수인지/값인지 여부로 알고있었는데, 강의의 경우에서 함수를 통해 그저 값만 반환하는 것 같아서 useCallback/useMemo가 갑자기 혼란스러웠는데, 이 경우에는 파라미터를 염두하기 위해 useCallback을 쓰는 것이군요. 답변 감사합니다!😊
클로드 초기 설정
0
2
1
사용자 스코프 설정 파일 적용 문제
0
5
1
클로드코드 유료플랜 할인 방법이 있을까요?
0
18
0
제네릭 클래스 핸드북 페이지가 undefined라고 나옵니다.
0
8
1
API Error : 400 에러의 원인과 해결방법이 궁금합니다!!
0
16
2
퍼미션 권한 설정 문의
0
18
2
Next.js + Tanstack Query BFF 구조 질문
0
12
1
커서에서 shift+enter가 안됩니다.
0
19
2
mcp 설치를 못하겠어요
0
33
2
라이브러리 관련 질문 있습니다!
0
24
2
output-styles은 Claude Code의 공식 기능이 아니라고 하는데 혹시 변경된걸까요?
0
23
2
클로드 변경
0
25
2
제네릭을 사용하실 때 콤마
0
167
2
popover createPortal 방식에서
0
147
2
스낵바를 만들 때 snackBarContext와 snackBarSetContext
0
146
2
snackbar를 createportal를 썻을 때 갯수 조절이 가능한가요?
0
193
1
createPortal 활용해서 modal 만들 때 활용한 MutationObserver 코드 관련 질문
0
356
3
강의자료 (보일러플레이트) 다운로드 관련 질문
0
256
1
모달만드는 방식 질문
0
243
1
캐러셀 동작원리
0
306
2
dropdown 내부에서 data의 타입을 알아야 할까요?
0
199
1
Provider를 외부에 노출하는 것보다 내부 로직으로 숨기는 것은 어떤가요?
0
288
2
keyEventMap를 전역에 구현하신 이유가 궁금합니다!
0
272
2
테일윈드로 포탈로 모달작성시 뒤에 클릭이 안됩니다
1
316
2

