useCallback 심도있게 살펴보기
useCallback 은 함수를 cache 로 저장 해서 re-render 할 때 사용할 수 있도록 하는 React Hook 이다.
아 슈밤 글 다썼는데 코드로 테스트하다가 다 날라갔다... ㅠㅠ
loop 3중첩으로 1000*1000*1000 으로 돌리니까 브라우저 바로 뻑나네.............
ProductPage가 리렌더링 되면 handleSubmit 함수는 새롭게 생성될까?
아니면 동일한 로직의 함수이니 알아서 이전 렌더링 때 만든 handleSubmit 함수를 그대로 사용할까?
정답은 함수를 새롭게 생성한다. 자바스크립트에서 function () {} 혹은 () => {} 인 경우에는 함수를 새로 생성하기 때문이다.
이런식으로 함수를 useCallback 으로 감싸고 dependency array 에 react value 들을 넣어주면 된다.
그러면 dependecy array 에 있는 value 들이 바뀔 때만 handleSubmit 을 재생성하고
바뀌지 않을 때는 이전 렌더링 때 생성한 handleSubmit 을 그대로 사용하여 최적화가 이루어진다.
위의 예시에서 todos 가 dependency array 에 있다.
setTodos 에서 todos 를 사용하기 때문이다.
그래서 handleAddTodo 가 수행되면 todos 가 변경되고, handleAddTodo 도 재생성된다.
위의 코드를 updater function 을 사용해 개선할 수 있다.
setTodos(todos => [...todos, newTodo]); 의 todos는 state 의 todos 가 아니다.
setTodos(prevState => [...prevState, newTodo]); 로 바꿀 수 있다.
즉 마치 함수형 프로그래밍을 하듯 함수로 바꿔주면
handleAddTodo 내에서 todos 를 사용하지 않게 되고
dependency 에서 todos 를 제거할 수 있다.
따라서 handleAddTodo 가 실행되어 todos 가 변경되어도 handleAddTodo 는 재생성하지 않게 되고
기존의 기능은 그대로 수행하기 때문에 더 좋은 최적화를 할 수 있다.
(즉, dependecy array 에는 최대한 적은 value 들이 있는게 좋다.)
공식 문서에는 없는 또다른 방법
import { useCallback } from 'react';
import ShippingForm from './ShippingForm.js';
function handleSubmit(orderDetails, productId, referrer) {
post('/product/' + productId + '/buy', {
referrer,
orderDetails,
});
}
export default function ProductPage({ productId, referrer, theme }) {
return (
<div className={theme}>
<ShippingForm onSubmit={handleSubmit} productId={productId} referrer={referrer} />
</div>
);
}
function post(url, data) {
// Imagine this sends a request...
console.log('POST /' + url);
console.log(data);
}
위의 코드처럼 handleSubmit 을 ProductPage 외부로 빼는 방법 가능하다.
이 경우 ProductPage가 리렌더링 되더라도 handleSubmit 은 재생성하지 않는다.
다만 ShippingForm 에 productId 와 referrer 를 전달하고 있는데
props drilling 이 발생하는 단점이 있겠다.
공식 문서에는 없지만 하나를 추리할 수 있다.
useCallback을 이용해 함수를 재생성하는 것을 막아 최적화를 할 수 있다 -> 즉, 함수를 생성하는 것은 오버헤드이다.
위의 예시는 간단한 함수라 useCallback 을 쓰냐 안쓰냐에 따라 큰 차이가 없겠지만
극단적인 예로 10000줄 짜리 함수라면 성능을 훨씬 좋게 만들 수 있을 것이다.
(물론 그런 함수는 만들지 않아야 겠지만)
useCallback을 많이 사용하면 브라우저 캐시나 메모리를 많이 사용하게 되진 않을까?
useCallback이 함수를 cache 에 저장하니까 useCallback을 마구 남발하면 그만큼 브라우저의 메모리를 많이 사용하게 되니까 성능에 악영향을 미치진 않을까
그런데 이와 관련한 내용은 공식 문서에 없었다.
크게 상관은 없는 것 같다.
회사 코드에 useCallback 적용하기
회사에 "릴레이 특가" 페이지에 1초마다 남은 시간이 1초씩 줄어드는 코드가 있었다.
함수가 복잡하진 않았지만 1초마다 실행되기 때문에
자주 실행된다고 생각하여 useCallback 을 사용해 최적화를 해줬다.
React.dev 문서에 따르면, 느려지는 부분이 느껴진다면 React Developer Tools의 profiler 로 확인하여 useCallback 이나 memo 를 활용하라고 적혀있다. 추후에도 이 툴을 이용해 최적화 작업 진행 해보려고 한다.