Advanced Hook ((better)) -
| Primitive Hook | Advanced Counterpart | Problem Solved | |----------------|----------------------|----------------| | useState | useReducer | Complex state transitions with interdependent logic | | Inline functions | useCallback | Unnecessary child re-renders due to function recreation | | Expensive computations | useMemo | Recalculating derived data on every render | | useEffect + DOM reads | useLayoutEffect | Visual flicker or layout shifts | | Prop drilling | useContext + useReducer | Global state management without Redux | | Forwarding refs | useImperativeHandle | Exposing limited imperative methods | When a component’s state involves multiple sub-values or transitions that depend on previous state, useState becomes verbose and error-prone. useReducer shines here. Example: Form with validation, submission, and error states const formReducer = (state, action) => switch (action.type) case 'FIELD_CHANGE': return ...state, [action.field]: action.value, error: null ; case 'SUBMIT_LOADING': return ...state, isLoading: true, error: null ; case 'SUBMIT_SUCCESS': return ...state, isLoading: false, isSuccess: true ; case 'SUBMIT_ERROR': return ...state, isLoading: false, error: action.error ; default: return state; ; function AdvancedForm() const [state, dispatch] = useReducer(formReducer, email: '', password: '', isLoading: false, error: null, isSuccess: false, ); // ... dispatch calls are self-documenting
Now go forth, abstract wisely, and may your dependency arrays always be complete.
const FancyInput = forwardRef((props, ref) => const inputRef = useRef(); useImperativeHandle(ref, () => ( focus: () => inputRef.current.focus(), clear: () => inputRef.current.value = ''; , shake: () => /* animation logic */ )); return <input ref=inputRef ...props />; ); // Parent usage: const ref = useRef(); <FancyInput ref=ref /> ref.current.shake(); advanced hook
This custom hook elegantly solves the "compare current vs previous" problem without extra state. Sometimes a parent component needs to call a child’s method (e.g., focus an input, reset a form, start an animation). While React discourages imperative code, advanced cases justify it.
Now any component can add debounced search with two lines of code. Custom hooks compose other advanced hooks, hiding complexity beautifully. Even advanced developers fall into traps: | Primitive Hook | Advanced Counterpart | Problem
Enter . These are not just obscure APIs; they are architectural patterns and built-in tools ( useReducer , useCallback , useMemo , useRef , useLayoutEffect , useImperativeHandle , useDebugValue , and custom hooks) that, when mastered, transform your code from "working" to elegant, predictable, and highly performant .
| Anti-Pattern | Why It’s Bad | Better Approach | |--------------|---------------|------------------| | Overly broad dependencies in useCallback | Memoization breaks on every render | Break logic into smaller hooks or use useReducer | | useMemo for simple arithmetic | Wastes memory and CPU | Let it compute; it’s cheap | | useLayoutEffect for data fetching | Blocks paint unnecessarily | useEffect + loading state | | Custom hook that causes cascade re-renders | Returns new object/array each call | Memoize returned object with useMemo | When building custom hooks, React DevTools shows generic "CustomHook" entries. Make them useful with useDebugValue : dispatch calls are self-documenting Now go forth, abstract
Now in DevTools, your hook displays FriendStatus: "Online" instead of an opaque value. Consider a dashboard that subscribes to a WebSocket stream, processes messages with expensive filtering, and exposes controls to pause/resume.