[React] 컴포넌트 성능 개선 React.memo()
1.React.memo란?
React.memo는 함수형 컴포넌트에서 동일한 파라미터로 생성된 함수들을 메모이제이션을 이용하여 재사용하게 도와 줍니다.
예를 들어 다음과 같은 상황이 있다고 가정해보겠습니다.
이름, 나이, 주소값을 받는 useState를 이용해 표현하고 있고 주소를 제외한 이름, 나이를 파라미터로 받는 컴포넌트가 있습니다.
여기서 이름, 나이 값이 변경되면 해당 값을 가져다 사용하는 컴포넌트도 당연히 리 렌더링이 발생되어 수정된 화면을 보여줘야 합니다.
하지만 주소값이 변경될 경우에는 어떨까요?
주소 값이 변경되더라도 useState로 정의된 값들을 같이 리 렌더링이 발생되기 때문에 이름,나이,값이 동일함에도 불구 하고 해당 값들을 가져다 사용하는 컴포넌트도 리 렌더링이 발생됩니다.
결국 굳이 리렌더링하지 않아도 되는 상황에서 리 렌더링을 함으로써 추가적인 작업을 해주게 됩니다.
이런 상황에 React.memo를 이용해줄 수 있습니다.
React.memo를 사용해 컴포넌트에서 전달받는 파라미터의 값을 확인하고 파라미터 값이 동일할 경우에는 리 렌더링을 발생시키지 않게 됩니다.
결과적으로 주소값이 변경되더라도 해당 컴포넌트는 리 렌더링이 발생되지 않고 단순하게 저장해둔 값을 가져다 사용함으로써 서비스 속도를 향상시켜줍니다.
하지만 React.memo에도 당연히 단점이 존재합니다.
메모이제이션 처리를 위한 작업 시간이 필요하기 때문에 만약 동일한 파라미터 값이 들어가는 경우가 드문 경우, 즉 메모이제이션 처리가 이루어진 컴포넌트가 이후에 자주 사용되지 않는 환경에서는 오히려 악영향을 미칠 수 있습니다.
무분별한 사용은 오히려 서비스 속도를 낮추는 결과를 초래하기 때문에 동일 파라미터가 자주 사용되는지를 잘 판단후에 사용해주어야 합니다.
무분별한 사용은 사용자에게 오히려 서비스 속도를 낮추는 결과를 초래하기에 동일 파라미터가 자주 사용되는지 잘 판단 후에 사용해주셔야 합니다.
React.memo 사용법은 간단합니다.
우선 'react'패키지에서 제공해주는 기능이기에 리액트를 개발하는 곳 어디에서든 사용하실 수 있습니다.
간단한 예시 코드를 작성해보겠습니다.
예시 코드
코드 실행은 위에서 언급한 예시를 토대로 동일한 결과를 나타내게 작성했습니다.
가장 먼저 App.jsx 부분 입니다.
다음과 같이 작성하면 되겠습니다.
import React, { useState } from 'react'
import MyComponent, {MemoizedMyComponent} from './MyComponent';
// import './App.css';
const App = () => {
const [name, setName] = useState('');
const [age, setAge] = useState('');
const [address, setAddress] = useState('');
return (
<div>
<table>
<tbody>
<tr>
<td>
<span>이름</span>
</td>
<td>
<input type="text" value={name} onChange={(e) => setName(e.target.value ? Number(e.target.value) : undefined) } />
</td>
</tr>
<tr>
<td>
<span>나이</span>
</td>
<td>
<input type="text" value={age} onChange={(e) => setAge(e.target.value ? Number(e.target.value) : undefined)} />
</td>
</tr>
<tr>
<td>
<span>주소</span>
</td>
<td>
<input type="text" value={address} onChange={(e) => setAddress(e.target.value ? Number(e.target.value) : undefined)} />
</td>
</tr>
<MyComponent name={name} age={age} /> {/* 메모이제이션이 적용되지 않는 컴포넌트 */}
<MemoizedMyComponent name={name} age={age} /> {/* 메모이제이션이 적용된 컴포넌트 */}
</tbody>
</table>
</div>
);
}
export default App;
그리고 App.jsx와 동일 위치에 myComponent.jsx를 작성해 다음과 같이 작성해보겠습니다.
import React from 'react'
const MyComponent = ({name, age}) => {
React.useEffect(() => {
console.log (`name: ${name}, age: ${age}`); // 리렌더링 될때마다 출력
});
return (
<div style={{borderTop: '2px solid orange', borderBottom: '2px solid orange', margin: '6px 0'}}>
<h2>name: {name}</h2>
<h4>Age: {age}</h4>
</div>
)
}
export default MyComponent // 메모이제이션 적용 X
export const MemoizedMyComponent = React.memo(MyComponent); // 메모이제이션 적용 O
코드를 작성하고 실행해보면
간단히 설명하자면
- 이름을 입력할 경우 -> 파라미터로 전달되는 값이어서 모든 컴포넌트가 리 렌더링 되어 콘솔에 출력(2개씩 출력)
- 나이를 입력할 경우 -> 이름과 동일
- 주소를 입력할 경우 -> 파라미터로 전달되는 값이 아니라 메모이제이션이 적용되지 않는 컴포넌트만 출력 (1개씩 출력)
참고자료