Rendering & Optimization in React

Overview

Rendering in React is the process of converting data (state, props) into UI.

Optimization ensures better performance and avoids unnecessary re-renders.

1. Conditional Rendering

Use if, &&, or ternary ? to render UI based on conditions.

// Using ternary
{isLoggedIn ? <Dashboard /> : <Login />}

// Using &&
{loading && <Spinner />}

Key Point:

  • Conditional rendering is pure JavaScript logic.
  • Don’t render hidden elements unnecessarily — it still affects performance.

2. List Rendering & Key

Render arrays of components using .map().

const users = ['Alice', 'Bob'];

return (
  <ul>
    {users.map((name, index) => (
      <li key={index}>{name}</li>
    ))}
  </ul>
);

Key Point:

  • key helps React track changes in lists.
  • Avoid using the index as the key if the list items can change order.

3. React.memo

React.memo is a higher-order component that memoizes a component, preventing re-render unless props change.

const UserCard = React.memo(function UserCard({ user }) {
  return <div>{user.name}</div>;
});

Key Point:

  • Use React.memo when:
    • Component renders often with same props
    • Component is pure and stateless
  • Combine with useCallback / useMemo for best result

4. useMemo

useMemo memoizes the result of a computation between renders.

const sortedData = useMemo(() => sortUsers(users), [users]);

Key Point:

  • Use for expensive computations.
  • Don’t overuse — it adds complexity.

5. useCallback

Memoizes a function to avoid re-creating it on each render.

const handleClick = useCallback(() => {
  doSomething();
}, [dependency]);

Use when:

  • You pass functions to memoized children (React.memo)
  • You want to optimize render of deeply nested components

6. Lazy Loading & Code Splitting

Split code and load lazily to improve initial load time.

const LazyComponent = React.lazy(() => import('./MyComponent'));

<Suspense fallback={<div>Loading...</div>}>
  <LazyComponent />
</Suspense>

Key Point:

  • Requires React.Suspense.
  • Improves performance in large apps.

Summary Table

TechniquePurposeWhen to Use
React.memoPrevents re-render if props haven’t changedFor pure functional components that re-render with the same props
useMemoMemoizes the result of an expensive calculationWhen you have heavy computations that depend on specific values
useCallbackMemoizes a function between rendersWhen passing callbacks to memoized child components (React.memo)
React.lazyLoads a component lazily (code splitting)When you want to reduce initial bundle size
SuspenseFallback UI while loading lazy componentsUsed with React.lazy or concurrent features
key in listsHelps React identify which items changed, added, or removedAlways use stable, unique keys (avoid index if list order may change)
Conditional renderingRenders elements based on conditionsWhen content should change depending on state or props

Best Practices

PracticeExplanation
Avoid unnecessary stateDon’t store values in state that can be derived from props or other state.
Use stable keysAlways use unique identifiers (like IDs) instead of array index as key.
Break down large componentsSplit big components into smaller, focused components.
Memoize wiselyUse React.memo, useMemo, useCallback only when profiling shows benefit.
Profile performanceUse React DevTools Profiler to detect unnecessary renders.
Defer non-critical componentsLoad rarely-used components with React.lazy and Suspense.
Minimize re-rendersAvoid prop changes or state updates that trigger re-renders unnecessarily.

Bình luận

Để lại một bình luận

Email của bạn sẽ không được hiển thị công khai. Các trường bắt buộc được đánh dấu *