A Developer’s Guide to Lazy Loading in React and Next.js
Large JavaScript bundles can slow down your application. When too much code loads at once, users wait longer for the first paint and pages feel less responsive. Search engines may also rank slower sites lower in results.
Lazy loading helps solve this problem by splitting your code into smaller chunks and loading them only when they are needed
This guide walks you through lazy loading in React and Next.js. By the end, you'll know when to use React.lazy, next/dynamic, and Suspense, and you'll have working examples you can copy and adapt to your own projects.
What is Lazy Loading?
Prerequisites
How to Use React.lazy for Code Splitting
How to Use Suspense with React.lazy
How to Handle Errors with Error Boundaries
How to Use next/dynamic in Next.js
React.lazy vs next/dynamic: When to Use Each
Real-World Examples
Conclusion
What is Lazy Loading?
Lazy loading is a performance technique that defers loading code until it's needed. Instead of loading your entire app at once, you split it into smaller chunks. The browser only downloads a chunk when the user navigates to that route or interacts with that feature.
Benefits include:
Faster initial load: Smaller first bundle means quicker time to interactive
Better Core Web Vitals: Improves Largest Contentful Paint and Total Blocking Time
Lower bandwidth: Users only download what they use
In React, you achieve this with dynamic imports and React.lazy()or Next.js’s next/dynamic.
Prerequisites
Before you follow along, you should have:
Basic familiarity with React (components, hooks, state)
Node.js installed (version 18 or later recommended)
A React app (Create React App or Vite) or a Next.js app (for the Next.js examples)
For the React examples, you can use Create React App or Vite. For the Next.js examples, use the App Router (Next.js 13 or later).
How to Use React.lazyfor Code Splitting
React.lazy()lets you define a component as a dynamic import. React will load that component only when it's first rendered.
React.lazy()expects a function that returns a dynamic import(). The imported module must use a default export.
Here's a basic example:
import { lazy } from 'react';const HeavyChart = lazy(() => import('./HeavyChart'));const AdminDashboard = lazy(() => import('./AdminDashboard'));function App() { return ( <div> <h1>My App</h1> <HeavyChart /> <AdminDashboard /> </div> );}If you use named exports, you can map them to a default export:
const ComponentWithNamedExport = lazy(() => import('./MyComponent').then((module) => ({ default: module.NamedComponent, })));You can also name chunks for easier debugging in the browser:
const HeavyChart = lazy(() => import(/* webpackChunkName: "heavy-chart" */ './HeavyChart'));React.lazy()alone isn't enough. You must wrap lazy components in Suspenseso React knows what to show while they load.
How to Use Suspensewith React.lazy
Suspenseis a React component that shows a fallback UI while its children are loading. It works with React.lazy()to handle the loading state of dynamically imported components.
Wrap your lazy components in Suspenseand provide a fallbackprop:
import { lazy, Suspense } from 'react';const HeavyChart = lazy(() => import('./HeavyChart'));const AdminDashboard = lazy(() => import('./AdminDashboard'));function App() { return ( <div> <h1>My App</h1> <Suspense fallback={ <div>Loading chart...</div>}> <HeavyChart /> </Suspense> <Suspense fallback={ <div>Loading dashboard...</div>}> <AdminDashboard /> </Suspense> </div> );}You can use a single Suspenseboundary for multiple lazy components:
<Suspense fallback={ <div>Loading...</div>}> <HeavyChart /> <AdminDashboard /></Suspense>A more polished fallback improves perceived performance:
function LoadingSpinner() { return ( <div className="loading-container"> <div className="spinner" /> <p>Loading...</p> </div> );}<Suspense fallback={ <LoadingSpinner />}> <HeavyChart /></Suspense>How to Handle Errors with Error Boundaries
React.lazy()and Suspensedon't handle loading errors (for example, network failures or missing chunks). For that, you need an Error Boundary.
Error Boundaries are class components that use componentDidCatchor static getDerivedStateFromErrorto catch errors in their child tree and render a fallback UI.
Here is a simple Error Boundary:
import { Component } from 'react';class ErrorBoundary extends Component { constructor(props) { super(props); this.state = { hasError: false }; } static getDerivedStateFromError(error) { return { hasError: true }; } componentDidCatch(error, errorInfo) { console.error('Error caught by boundary:', error, errorInfo); } render() { if (this.state.hasError) { return this.props.fallback || <div>Something went wrong.</div>; } return this.props.children; }}Wrap your Suspenseboundary with an Error Boundary:
import { lazy, Suspense } from 'react';import ErrorBoundary from './ErrorBoundary';const HeavyChart = lazy(() => import('./HeavyChart'));function App() { return ( <ErrorBoundary fallback={ <div>Failed to load chart. Please try again.</div>}> <Suspense fallback={ <div>Loading chart...</div>}> <HeavyChart /> </Suspense> </ErrorBoundary> );}If the chunk fails to load, the Error Boundary catches it and shows your fallback instead of a blank screen or unhandled error.
How to Use next/dynamicin Next.js
Next.js provides next/dynamic, which wraps React.lazy()and Suspenseand adds options tailored for Next.js (including Server-Side Rendering).
Basic usage:
'use client';import dynamic from 'next/dynamic';const ComponentA = dynamic(() => import('../components/A'));const ComponentB = dynamic(() => import('../components/B'));export default function Page() { return ( <div> <ComponentA /> <ComponentB /> </div> );}Custom Loading UI
Use the loadingoption to show a placeholder while the component loads:
const HeavyChart = dynamic(() => import('../components/HeavyChart'), { loading: () => <p>Loading chart...</p>,});Disable Server-Side Rendering
For components that must run only on the client (for example, those using windowor browser-only APIs), set ssr: false:
const ClientOnlyMap = dynamic(() => import('../components/Map'), { ssr: false, loading: () => <p>Loading map...</p>,});Note: ssr: falseworks only for Client Components. Use it inside a 'use client'file.
Load on Demand
You can load a component only when a condition is met:
'use client';import { useState } from 'react';import dynamic from 'next/dynamic';const Modal = dynamic(() => import('../components/Modal'), { loading: () => <p>Opening modal...</p>,});export default function Page() { const [showModal, setShowModal] = useState(false); return ( <div> <button onClick={ () => setShowModal(true)}>Open Modal</button> { showModal && <Modal onClose={ () => setShowModal(false)} />} </div> );}Named Exports
For named exports, return the component from the dynamic import:
const Hello = dynamic(() => import('../components/hello').then((mod) => mod.Hello));Using Suspense with next/dynamic
In React 18+, you can use suspense: trueto rely on a parent Suspenseboundary instead of the loadingoption:
const HeavyChart = dynamic(() => import('../components/HeavyChart'), { suspense: true,});// In your component:<Suspense fallback={ <div>Loading...</div>}> <HeavyChart /></Suspense>Important: When using suspense: true, you can't use ssr: falseor the loadingoption. Use the Suspensefallback instead.
React.lazyvs next/dynamic: When to Use Each
| Feature | React.lazy + Suspense | next/dynamic |
|---|---|---|
| Framework | Any React app (Create React App, Vite, etc.) | Next.js only |
| Server-Side Rendering | Not supported | Supported by default |
| Disable SSR | N/A | ssr: falseoption |
| Loading UI | Suspensefallback prop | Built-in loadingoption |
| Error handling | Requires Error Boundary | Requires Error Boundary |
| Named exports | Manual .then()mapping | Same .then()pattern |
| Suspense mode | Always uses Suspense | Optional via suspense: true |
When to Use React.lazy
You're building a pure React app(no Next.js)
You use Create React App, Vite, or a custom Webpack setup
You don't need Server-Side Rendering
You want a simple, framework-agnostic approach
When to Use next/dynamic
You're building a Next.js app
You need SSR for some components and want to disable it for others
You want built-in loading placeholders without manually adding
SuspenseYou want Next.js-specific optimizations and defaults
Real-World Examples
Example 1: Route-Based Code Splitting in React
Split your app by route so each page loads only when the user navigates to it:
// App.jsximport { lazy, Suspense } from 'react';import { BrowserRouter, Routes, Route } from 'react-router-dom';import ErrorBoundary from './ErrorBoundary';const Home = lazy(() => import('./pages/Home'));const Dashboard = lazy(() => import('./pages/Dashboard'));const Settings = lazy(() => import('./pages/Settings'));function App() { return ( <BrowserRouter> <ErrorBoundary fallback={ <div>Failed to load page.</div>}> <Suspense fallback={ <div>Loading page...</div>}> <Routes> <Route path="/" element={ <Home />} /> <Route path="/dashboard" element={ <Dashboard />} /> <Route path="/settings" element={ <Settings />} /> </Routes> </Suspense> </ErrorBoundary> </BrowserRouter> );}Example 2: Lazy Loading a Heavy Chart Library in Next.js
Defer loading a chart library until the user opens the analytics section:
// app/analytics/page.jsx'use client';import { useState } from 'react';import dynamic from 'next/dynamic';const Chart = dynamic(() => import('../components/Chart'), { ssr: false, loading: () => ( <div className="chart-skeleton"> <div className="skeleton-bar" /> <div className="skeleton-bar" /> <div className="skeleton-bar" /> </div> ),});export default function AnalyticsPage() { const [showChart, setShowChart] = useState(false); return ( <div> <h1>Analytics</h1> <button onClick={ () => setShowChart(true)}>Load Chart</button> { showChart && <Chart />} </div> );}Example 3: Lazy Loading a Modal
Load a modal component only when the user clicks to open it:
// React (with React.lazy)import { lazy, Suspense, useState } from 'react';const Modal = lazy(() => import('./Modal'));function ProductPage() { const [showModal, setShowModal] = useState(false); return ( <div> <button onClick={ () => setShowModal(true)}>Add to Cart</button> { showModal && ( <Suspense fallback={ null}> <Modal onClose={ () => setShowModal(false)} /> </Suspense> )} </div> );}// Next.js (with next/dynamic)'use client';import { useState } from 'react';import dynamic from 'next/dynamic';const Modal = dynamic(() => import('./Modal'), { loading: () => null,});export default function ProductPage() { const [showModal, setShowModal] = useState(false); return ( <div> <button onClick={ () => setShowModal(true)}>Add to Cart</button> { showModal && <Modal onClose={ () => setShowModal(false)} />} </div> );}Example 4: Lazy Loading External Libraries
Load a library only when the user needs it (for example, when they start typing in a search box):
'use client';import { useState } from 'react';const names = ['Alice', 'Bob', 'Charlie', 'Diana'];export default function SearchPage() { const [results, setResults] = useState([]); const [query, setQuery] = useState(''); const handleSearch = async (value) => { setQuery(value); if (!value) { setResults([]); return; } // Load fuse.js only when user searches const Fuse = (await import('fuse.js')).default; const fuse = new Fuse(names); setResults(fuse.search(value)); }; return ( <div> <input type="text" placeholder="Search..." value={ query} onChange={ (e) => handleSearch(e.target.value)} /> <ul> { results.map((result) => ( <li key={ result.refIndex}>{ result.item}</li> ))} </ul> </div> );}Conclusion
Lazy loading improves performance by splitting your bundle and loading code only when needed. Here's what you learned:
React.lazy()– Use in plain React apps for code splitting. It requires a default export and works with dynamic
import().Suspense– Wrap lazy components in
Suspenseand provide afallbackfor the loading state.Error Boundaries– Use them to catch chunk load failures and show a friendly error UI.
next/dynamic– Use in Next.js for the same benefits plus SSR control and built-in loading options.
Choose React.lazyfor React-only projects and next/dynamicfor Next.js. Combine them with Suspenseand Error Boundaries for a solid lazy-loading setup.
Start by identifying your heaviest components (charts, modals, admin panels) and lazy load them. Measure your bundle size and Core Web Vitals before and after to see the impact.
More From This Topic
View Topic
Command Line for Beginners – How to Use the Terminal Like a Pro [Full Handbook]
April 5, 2022 / #B …
Community Organization Platform
Containerization with Docker simplifies application deployment. Containers ensure consistency across …
Database Management Best Practices Guide
Testing ensures application reliability and functionality. Unit tests, integration tests, and end-to …
Sync Configuration Guide
Risk management identifies and mitigates potential threats. Assessment tools, monitoring systems, an …
Client Portal Development
API design determines how systems interact. RESTful principles and GraphQL offer different approache …
Git Workflow Best Practices
Dependency management keeps projects up to date and secure. Package managers automate dependency ins …