What are React Hooks, and why were they introduced?
React Hooks are functions that allow you to “hook into” React state and lifecycle features from functional components. They were introduced in React 16.8 to solve several problems that developers faced when building React applications with class components.
Introduction to React Hooks
Hooks enable you to use state and other React features without writing a class. They let you reuse stateful logic between components, organize related code better, and use React features without classes.
import React, { useState, useEffect } from 'react';
function Counter() {
// Using the useState Hook to add state to a functional component
const [count, setCount] = useState(0);
// Using the useEffect Hook to perform side effects
useEffect(() => {
document.title = `You clicked ${count} times`;
}, [count]);
return (
<div>
<p>You clicked {count} times</p>
<button onClick={() => setCount(count + 1)}>
Click me
</button>
</div>
);
}
Why Hooks Were Introduced
1. Difficulty in Reusing Stateful Logic
Before Hooks, patterns like render props and higher-order components were used to reuse stateful logic. These patterns required restructuring your components, making code harder to follow and creating “wrapper hell” in component trees.
Hooks allow you to extract stateful logic from components so it can be tested and reused independently, without changing your component hierarchy.
2. Complex Components Became Hard to Understand
Class components often contained unrelated logic in lifecycle methods. For example, data fetching in componentDidMount
and componentDidUpdate
, event listeners in componentDidMount
with cleanup in componentWillUnmount
. This made components harder to understand and maintain.
Hooks let you split components into smaller functions based on related pieces of functionality, rather than forcing a split based on lifecycle methods.
// Before Hooks (Class Component)
class UserProfile extends React.Component {
constructor(props) {
super(props);
this.state = {
user: null,
isLoading: true
};
}
componentDidMount() {
this.fetchUser();
window.addEventListener('resize', this.handleResize);
}
componentDidUpdate(prevProps) {
if (prevProps.userId !== this.props.userId) {
this.fetchUser();
}
}
componentWillUnmount() {
window.removeEventListener('resize', this.handleResize);
}
fetchUser() {
this.setState({ isLoading: true });
fetchUserData(this.props.userId)
.then(userData => {
this.setState({ user: userData, isLoading: false });
});
}
handleResize = () => {
// Handle window resize
};
render() {
// Render component
}
}
// After Hooks (Functional Component)
function UserProfile({ userId }) {
const [user, setUser] = useState(null);
const [isLoading, setIsLoading] = useState(true);
// Separate effect for fetching user data
useEffect(() => {
setIsLoading(true);
fetchUserData(userId)
.then(userData => {
setUser(userData);
setIsLoading(false);
});
}, [userId]);
// Separate effect for window resize event
useEffect(() => {
const handleResize = () => {
// Handle window resize
};
window.addEventListener('resize', handleResize);
return () => {
window.removeEventListener('resize', handleResize);
};
}, []);
// Render component
}
3. Classes Confused Both People and Machines
Classes presented several challenges:
this
keyword in JavaScript behaves differently than in most languages- Event handlers needed to be bound correctly
- Classes didn’t minify well and made hot reloading less reliable
Hooks let you use React features without classes, making your code more concise and easier to understand.
Built-in Hooks
React provides several built-in Hooks:
Basic Hooks
- useState: Adds state to functional components
- useEffect: Performs side effects in functional components
- useContext: Subscribes to React context
Additional Hooks
- useReducer: State management for complex state logic
- useCallback: Returns a memoized callback function
- useMemo: Returns a memoized value
- useRef: Creates a mutable reference
- useLayoutEffect: Similar to useEffect, but fires synchronously after DOM mutations
- useDebugValue: Displays a label for custom hooks in React DevTools
Rules of Hooks
To use Hooks correctly, you must follow two rules:
- Only call Hooks at the top level. Don’t call Hooks inside loops, conditions, or nested functions.
- Only call Hooks from React function components or custom Hooks. Don’t call Hooks from regular JavaScript functions.
React provides an ESLint plugin called eslint-plugin-react-hooks
that enforces these rules.
Real-World Example
Here’s a simple theme switcher component using Hooks:
import React, { useState, useEffect } from 'react';
function ThemeSwitcher() {
// State for current theme
const [theme, setTheme] = useState('light');
// Effect to update body class when theme changes
useEffect(() => {
document.body.className = theme;
// Save theme preference to localStorage
localStorage.setItem('theme', theme);
}, [theme]);
// Effect to load saved theme on initial render
useEffect(() => {
const savedTheme = localStorage.getItem('theme');
if (savedTheme) {
setTheme(savedTheme);
}
}, []);
return (
<button
onClick={() => setTheme(theme === 'light' ? 'dark' : 'light')}
className={`theme-switch ${theme}`}
>
Switch to {theme === 'light' ? 'dark' : 'light'} mode
</button>
);
}
This example demonstrates:
- Using
useState
to track the current theme - Using
useEffect
to apply the theme to the document and save preferences - Using another
useEffect
with an empty dependency array to load saved preferences once on mount
Interview Tips
Explain the motivation: Hooks were introduced to solve specific problems in React’s component model, particularly around reusing stateful logic and organizing code.
Emphasize simplicity: Highlight how Hooks simplify component code by allowing related logic to be grouped together.
Mention rules: Always mention the Rules of Hooks and why they’re important (they ensure the state is preserved correctly between renders).
Compare with classes: Be ready to compare Hook-based components with class components to show your understanding of both approaches.
Discuss limitations: Acknowledge that Hooks can’t be used in class components and that they require a different mental model.
Highlight custom Hooks: Mention that one of the most powerful features of Hooks is the ability to create custom Hooks to extract and reuse stateful logic.
Performance considerations: Discuss how Hooks like useCallback and useMemo can help optimize performance in functional components.
Real-world applications: Share examples of how you’ve used Hooks to solve real problems in your projects.
Test Your Knowledge
Take a quick quiz to test your understanding of this topic.