useCallback Hook in React
What is useCallback?
useCallback
is a React hook that returns a memoized callback function. It only changes if one of its dependencies changes. This is useful when passing callbacks to optimized child components that rely on reference equality to prevent unnecessary renders.
const memoizedCallback = useCallback(
() => {
doSomething(a, b);
},
[a, b], // dependencies array
);
Purpose of useCallback
The primary purpose of useCallback
is to optimize performance by:
- Preventing unnecessary re-creation of function references
- Avoiding unnecessary re-renders of child components that receive callbacks as props
- Maintaining referential equality between renders
Basic Syntax and Usage
import React, { useState, useCallback } from 'react';
function Counter() {
const [count, setCount] = useState(0);
const [otherState, setOtherState] = useState(0);
// Without useCallback - new function reference on every render
const incrementWithoutCallback = () => {
setCount(c => c + 1);
};
// With useCallback - same function reference unless dependencies change
const incrementWithCallback = useCallback(() => {
setCount(c => c + 1);
}, []); // Empty dependency array means function never changes
return (
<div>
<p>Count: {count}</p>
<p>Other state: {otherState}</p>
<button onClick={incrementWithCallback}>
Increment with useCallback
</button>
<button onClick={() => setOtherState(s => s + 1)}>
Update other state
</button>
{/* Child component receives the callback */}
<ButtonComponent onClick={incrementWithCallback} />
</div>
);
}
// Memoized child component that only re-renders when props change
const ButtonComponent = React.memo(function ButtonComponent({ onClick }) {
console.log('ButtonComponent rendered');
return <button onClick={onClick}>Click me</button>;
});
When to Use useCallback
1. When Passing Callbacks to Memoized Child Components
The most common use case for useCallback
is when passing callbacks to child components that are optimized with React.memo
:
function ParentComponent() {
const [todos, setTodos] = useState([]);
const [filter, setFilter] = useState('all');
// Memoize the callback to maintain reference equality
const handleAddTodo = useCallback((text) => {
const newTodo = { id: Date.now(), text, completed: false };
setTodos(prevTodos => [...prevTodos, newTodo]);
}, []); // No dependencies, so function reference never changes
const handleToggleTodo = useCallback((id) => {
setTodos(prevTodos =>
prevTodos.map(todo =>
todo.id === id ? { ...todo, completed: !todo.completed } : todo
)
);
}, []); // No dependencies, so function reference never changes
// Filter todos based on current filter
const filteredTodos = useMemo(() => {
switch (filter) {
case 'active': return todos.filter(todo => !todo.completed);
case 'completed': return todos.filter(todo => todo.completed);
default: return todos;
}
}, [todos, filter]);
return (
<div>
<AddTodoForm onAddTodo={handleAddTodo} />
<FilterButtons filter={filter} onFilterChange={setFilter} />
<TodoList todos={filteredTodos} onToggleTodo={handleToggleTodo} />
</div>
);
}
// Memoized components that only re-render when props change
const AddTodoForm = React.memo(function AddTodoForm({ onAddTodo }) {
// Component implementation
});
const TodoList = React.memo(function TodoList({ todos, onToggleTodo }) {
// Component implementation
});
2. When a Callback is a Dependency in Other Hooks
When a function is used as a dependency in useEffect
, useMemo
, or another useCallback
:
function SearchComponent({ onSearch }) {
const [query, setQuery] = useState('');
// Memoize the debounced search function
const debouncedSearch = useCallback(
debounce((searchTerm) => {
onSearch(searchTerm);
}, 500),
[onSearch] // Re-create when onSearch changes
);
// Use the memoized callback in useEffect
useEffect(() => {
if (query) {
debouncedSearch(query);
}
}, [query, debouncedSearch]); // debouncedSearch is a dependency
return (
<input
type="text"
value={query}
onChange={e => setQuery(e.target.value)}
placeholder="Search..."
/>
);
}
3. When Working with Custom Hooks
When creating custom hooks that return callbacks:
function useCounter(initialValue = 0) {
const [count, setCount] = useState(initialValue);
// Memoize these functions so consumers don't get new references
const increment = useCallback(() => {
setCount(c => c + 1);
}, []);
const decrement = useCallback(() => {
setCount(c => c - 1);
}, []);
const reset = useCallback(() => {
setCount(initialValue);
}, [initialValue]); // Re-create when initialValue changes
return { count, increment, decrement, reset };
}
// Usage in a component
function CounterComponent() {
const { count, increment, decrement, reset } = useCounter(0);
return (
<div>
<p>Count: {count}</p>
<button onClick={increment}>+</button>
<button onClick={decrement}>-</button>
<button onClick={reset}>Reset</button>
</div>
);
}
Dependencies Array
The second argument to useCallback
is the dependencies array:
const memoizedCallback = useCallback(
() => {
doSomething(a, b);
},
[a, b], // Only re-create callback if a or b changes
);
- Empty array
[]
: Callback is created once and never changes - With dependencies
[a, b]
: Callback is re-created when any dependency changes - No array provided: Callback is re-created on every render (defeats the purpose)
Common Pitfalls
1. Missing Dependencies
Forgetting to include values used inside the callback:
// BAD: Missing dependency
const [count, setCount] = useState(0);
const handleClick = useCallback(() => {
console.log(`Clicked ${count} times`);
// count is used but not listed as a dependency
}, []); // ESLint will warn about this
// GOOD: Include all dependencies
const handleClick = useCallback(() => {
console.log(`Clicked ${count} times`);
}, [count]); // Re-creates function when count changes
2. Unnecessary useCallback
Using useCallback
when it doesn’t provide any benefit:
// Unnecessary - no child components rely on reference equality
function SimpleComponent() {
// This doesn't help performance
const handleClick = useCallback(() => {
console.log('Clicked');
}, []);
return <button onClick={handleClick}>Click me</button>;
}
3. Inline Function Dependencies
Creating new functions inside useCallback
:
// BAD: Creates a new formatter function on every render
const formatData = useCallback((data) => {
// This creates a new function reference on every render
const formatter = (item) => `${item.name}: ${item.value}`;
return data.map(formatter);
}, []);
// GOOD: Move the formatter outside or memoize it
const formatter = useCallback(
(item) => `${item.name}: ${item.value}`,
[]
);
const formatData = useCallback((data) => {
return data.map(formatter);
}, [formatter]);
useCallback vs. useMemo
useCallback(fn, deps)
returns a memoized callback functionuseMemo(() => fn, deps)
returns a memoized value (the result of calling the function)
// useCallback memoizes the function itself
const memoizedFunction = useCallback(() => {
return expensiveComputation(a, b);
}, [a, b]);
// useMemo memoizes the result of the function
const memoizedResult = useMemo(() => {
return expensiveComputation(a, b);
}, [a, b]);
// Equivalent to:
// const memoizedFunction = useMemo(() => () => expensiveComputation(a, b), [a, b]);
Best Practices
- Use with React.memo: Combine
useCallback
withReact.memo
for optimized child components - Include all dependencies: Always include all values from the component scope that the callback uses
- Create event handlers inline for simple components without memoized children
- Extract complex callbacks outside the component when they don’t depend on props or state
- Use the ESLint plugin for React Hooks to catch missing dependencies
Interview Tips
- Explain that
useCallback
is primarily a performance optimization, not necessary for correctness - Discuss how it works with React’s reconciliation process and memoization
- Mention that premature optimization with
useCallback
can sometimes hurt performance - Demonstrate understanding of the relationship between
useCallback
andReact.memo
- Be prepared to discuss when
useCallback
provides real benefits vs. when it’s unnecessary - Explain how
useCallback
differs fromuseMemo
and when to use each
Test Your Knowledge
Take a quick quiz to test your understanding of this topic.