| name | performance-optimization |
| description | This skill should be used when optimizing application performance - profiling, caching, lazy loading, code splitting, image optimization, and CDN strategies. |
Performance Optimization
Overview
Performance optimization is the systematic process of improving application speed, responsiveness, and resource efficiency. The core principle is to deliver the best user experience by minimizing load times, reducing time-to-interactive, and ensuring smooth interactions across all devices and network conditions.
Key Philosophy: Measure first, optimize second. Never optimize based on assumptions - always profile, measure, and validate improvements with real metrics.
When to Use
Apply performance optimization when:
- Initial page load exceeds 3 seconds
- Core Web Vitals fail (LCP > 2.5s, FID > 100ms, CLS > 0.1)
- Bundle sizes exceed 200KB gzipped for initial load
- Users report slow interactions or laggy UI
- Mobile performance significantly lags desktop
- Server response times exceed 200ms for API calls
- Database queries take longer than 100ms
- Images are not optimized or properly sized
- Render-blocking resources delay page display
- Memory leaks cause performance degradation over time
- High bounce rates correlate with slow load times
Performance Measurement
Core Web Vitals
Largest Contentful Paint (LCP): Measures loading performance
- Target: < 2.5 seconds
- Good: 0-2.5s | Needs Improvement: 2.5-4.0s | Poor: > 4.0s
- Measures when the largest content element becomes visible
- Optimize by: reducing server response time, optimizing images, eliminating render-blocking resources
First Input Delay (FID): Measures interactivity
- Target: < 100 milliseconds
- Good: 0-100ms | Needs Improvement: 100-300ms | Poor: > 300ms
- Measures time from first user interaction to browser response
- Optimize by: reducing JavaScript execution time, code splitting, web workers
Cumulative Layout Shift (CLS): Measures visual stability
- Target: < 0.1
- Good: 0-0.1 | Needs Improvement: 0.1-0.25 | Poor: > 0.25
- Measures unexpected layout shifts during page load
- Optimize by: setting explicit dimensions, avoiding dynamic content injection
Performance Budgets
Establish clear budgets for critical metrics:
// performance-budget.config.ts
export const PERFORMANCE_BUDGET = {
// Size budgets
maxBundleSize: 200 * 1024, // 200KB gzipped
maxInitialJS: 150 * 1024, // 150KB gzipped
maxInitialCSS: 30 * 1024, // 30KB gzipped
maxImageSize: 500 * 1024, // 500KB per image
// Timing budgets
maxLCP: 2500, // 2.5 seconds
maxFID: 100, // 100 milliseconds
maxTTI: 3800, // 3.8 seconds (Time to Interactive)
maxFCP: 1800, // 1.8 seconds (First Contentful Paint)
// Resource budgets
maxRequests: 50, // Total HTTP requests
maxFonts: 2, // Number of font families
maxThirdParty: 3, // Third-party scripts
// API budgets
maxAPIResponse: 200, // 200ms for API calls
maxDBQuery: 100, // 100ms for database queries
} as const;
// Monitor in CI/CD
export function validateBudget(metrics: PerformanceMetrics): boolean {
const violations: string[] = [];
if (metrics.bundleSize > PERFORMANCE_BUDGET.maxBundleSize) {
violations.push(`Bundle size ${metrics.bundleSize} exceeds ${PERFORMANCE_BUDGET.maxBundleSize}`);
}
if (metrics.lcp > PERFORMANCE_BUDGET.maxLCP) {
violations.push(`LCP ${metrics.lcp}ms exceeds ${PERFORMANCE_BUDGET.maxLCP}ms`);
}
if (violations.length > 0) {
console.error('Performance Budget Violations:', violations);
return false;
}
return true;
}
Profiling Tools
Chrome DevTools Performance Panel:
// Enable performance profiling
performance.mark('component-render-start');
// ... component render logic
performance.mark('component-render-end');
performance.measure('component-render', 'component-render-start', 'component-render-end');
// Log measurements
const measures = performance.getEntriesByType('measure');
console.table(measures.map(m => ({
name: m.name,
duration: `${m.duration.toFixed(2)}ms`
})));
Lighthouse CI Integration:
# .github/workflows/lighthouse.yml
name: Lighthouse CI
on: [pull_request]
jobs:
lighthouse:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Run Lighthouse CI
run: |
npm install -g @lhci/cli
lhci autorun
Web Vitals Monitoring:
// lib/web-vitals.ts
import { onCLS, onFID, onLCP, onFCP, onTTFB } from 'web-vitals';
export function reportWebVitals() {
onCLS((metric) => sendToAnalytics('CLS', metric.value));
onFID((metric) => sendToAnalytics('FID', metric.value));
onLCP((metric) => sendToAnalytics('LCP', metric.value));
onFCP((metric) => sendToAnalytics('FCP', metric.value));
onTTFB((metric) => sendToAnalytics('TTFB', metric.value));
}
function sendToAnalytics(name: string, value: number) {
// Send to your analytics service
if (typeof window.gtag !== 'undefined') {
window.gtag('event', name, {
value: Math.round(name === 'CLS' ? value * 1000 : value),
event_category: 'Web Vitals',
non_interaction: true,
});
}
}
Frontend Optimization
Code Splitting and Lazy Loading
Route-based Code Splitting:
// BEFORE: Everything loaded upfront (800KB bundle)
import Home from './pages/Home';
import Dashboard from './pages/Dashboard';
import Profile from './pages/Profile';
import Settings from './pages/Settings';
// AFTER: Lazy load routes (200KB initial, rest on-demand)
import { lazy, Suspense } from 'react';
const Home = lazy(() => import('./pages/Home'));
const Dashboard = lazy(() => import('./pages/Dashboard'));
const Profile = lazy(() => import('./pages/Profile'));
const Settings = lazy(() => import('./pages/Settings'));
function App() {
return (
<Suspense fallback={<LoadingSpinner />}>
<Routes>
<Route path="/" element={<Home />} />
<Route path="/dashboard" element={<Dashboard />} />
<Route path="/profile" element={<Profile />} />
<Route path="/settings" element={<Settings />} />
</Routes>
</Suspense>
);
}
// Result: Initial bundle reduced by 75%, FCP improved from 4.2s to 1.5s
Component-level Code Splitting:
// components/HeavyChart.tsx - Split large dependencies
import { lazy, Suspense } from 'react';
// BEFORE: Chart library loaded always (300KB)
import { Chart } from 'chart.js';
// AFTER: Load only when needed
const Chart = lazy(() => import('chart.js').then(module => ({
default: module.Chart
})));
export function AnalyticsDashboard({ data }) {
const [showChart, setShowChart] = useState(false);
return (
<div>
<button onClick={() => setShowChart(true)}>
Show Chart
</button>
{showChart && (
<Suspense fallback={<ChartSkeleton />}>
<Chart data={data} />
</Suspense>
)}
</div>
);
}
Image Optimization
Modern Image Formats and Lazy Loading:
// components/OptimizedImage.tsx
import { useState, useEffect } from 'react';
interface OptimizedImageProps {
src: string;
alt: string;
width: number;
height: number;
priority?: boolean;
}
export function OptimizedImage({
src,
alt,
width,
height,
priority = false
}: OptimizedImageProps) {
const [isLoaded, setIsLoaded] = useState(false);
// Generate srcset for responsive images
const srcSet = `
${src}?w=${width}&q=75&fm=webp 1x,
${src}?w=${width * 2}&q=75&fm=webp 2x
`;
return (
<picture>
{/* Modern formats */}
<source type="image/avif" srcSet={`${src}?w=${width}&fm=avif`} />
<source type="image/webp" srcSet={srcSet} />
{/* Fallback */}
<img
src={`${src}?w=${width}&q=75`}
srcSet={srcSet}
alt={alt}
width={width}
height={height}
loading={priority ? 'eager' : 'lazy'}
decoding={priority ? 'sync' : 'async'}
onLoad={() => setIsLoaded(true)}
style={{
objectFit: 'cover',
backgroundColor: '#f0f0f0',
transition: 'opacity 0.3s',
opacity: isLoaded ? 1 : 0
}}
/>
</picture>
);
}
// Result: 60-80% smaller images, faster loading
// JPEG (500KB) -> WebP (150KB) -> AVIF (100KB)
Next.js Image Component:
// BEFORE: Unoptimized images
<img src="/hero.jpg" alt="Hero" width={1200} height={600} />
// AFTER: Automatic optimization
import Image from 'next/image';
<Image
src="/hero.jpg"
alt="Hero"
width={1200}
height={600}
priority // For above-the-fold images
placeholder="blur"
blurDataURL="..." // Low-quality placeholder
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
/>
// Automatically:
// - Serves WebP/AVIF when supported
// - Generates multiple sizes
// - Lazy loads by default
// - Prevents CLS with aspect ratio
CSS Optimization
Critical CSS Extraction:
// next.config.js
module.exports = {
experimental: {
optimizeCss: true, // Enable CSS optimization
},
};
// Extract critical CSS
import { CriticalCSS } from 'critters';
export async function extractCriticalCSS(html: string) {
const critters = new CriticalCSS({
path: 'public',
logLevel: 'silent',
preload: 'swap',
pruneSource: true,
});
return await critters.process(html);
}
// Result: FCP improved from 2.8s to 1.2s
CSS-in-JS Optimization:
// BEFORE: Runtime CSS generation (slow)
import styled from 'styled-components';
const Button = styled.button`
padding: 12px 24px;
background: blue;
color: white;
`;
// AFTER: Zero-runtime CSS with Linaria
import { styled } from '@linaria/react';
const Button = styled.button`
padding: 12px 24px;
background: blue;
color: white;
`;
// Or use CSS Modules for best performance
import styles from './Button.module.css';
<button className={styles.button}>Click</button>
JavaScript Bundle Optimization
Tree Shaking and Dead Code Elimination:
// BEFORE: Importing entire library (300KB)
import _ from 'lodash';
const result = _.debounce(fn, 300);
// AFTER: Import only what you need (5KB)
import debounce from 'lodash/debounce';
const result = debounce(fn, 300);
// Even better: Use native or lightweight alternatives
function debounce<T extends (...args: any[]) => any>(
fn: T,
delay: number
): (...args: Parameters<T>) => void {
let timeoutId: NodeJS.Timeout;
return (...args) => {
clearTimeout(timeoutId);
timeoutId = setTimeout(() => fn(...args), delay);
};
}
Webpack Bundle Analyzer:
// webpack.config.js
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
module.exports = {
plugins: [
new BundleAnalyzerPlugin({
analyzerMode: 'static',
openAnalyzer: false,
reportFilename: 'bundle-report.html',
}),
],
optimization: {
splitChunks: {
chunks: 'all',
cacheGroups: {
vendor: {
test: /[\\/]node_modules[\\/]/,
name: 'vendors',
priority: 10,
},
common: {
minChunks: 2,
priority: 5,
reuseExistingChunk: true,
},
},
},
},
};
Caching Strategies
Browser Caching
Cache-Control Headers:
// next.config.js - Static assets
module.exports = {
async headers() {
return [
{
source: '/static/:path*',
headers: [
{
key: 'Cache-Control',
value: 'public, max-age=31536000, immutable', // 1 year
},
],
},
{
source: '/:path*',
headers: [
{
key: 'Cache-Control',
value: 'public, max-age=0, must-revalidate', // Always revalidate HTML
},
],
},
];
},
};
Local Storage Caching:
// lib/cache.ts
interface CacheItem<T> {
data: T;
timestamp: number;
ttl: number;
}
export class BrowserCache {
private prefix = 'app_cache_';
set<T>(key: string, data: T, ttl: number = 3600000): void {
const item: CacheItem<T> = {
data,
timestamp: Date.now(),
ttl,
};
try {
localStorage.setItem(
`${this.prefix}${key}`,
JSON.stringify(item)
);
} catch (error) {
console.warn('Cache set failed:', error);
}
}
get<T>(key: string): T | null {
try {
const item = localStorage.getItem(`${this.prefix}${key}`);
if (!item) return null;
const cached: CacheItem<T> = JSON.parse(item);
const isExpired = Date.now() - cached.timestamp > cached.ttl;
if (isExpired) {
this.delete(key);
return null;
}
return cached.data;
} catch (error) {
console.warn('Cache get failed:', error);
return null;
}
}
delete(key: string): void {
localStorage.removeItem(`${this.prefix}${key}`);
}
clear(): void {
Object.keys(localStorage)
.filter(key => key.startsWith(this.prefix))
.forEach(key => localStorage.removeItem(key));
}
}
// Usage
const cache = new BrowserCache();
async function fetchUserData(userId: string) {
const cached = cache.get<User>(`user_${userId}`);
if (cached) return cached;
const data = await fetch(`/api/users/${userId}`).then(r => r.json());
cache.set(`user_${userId}`, data, 300000); // 5 minutes
return data;
}
// Result: Reduced API calls by 70%, improved response time from 200ms to 5ms
Service Workers and PWA
Service Worker Caching Strategy:
// public/service-worker.ts
const CACHE_NAME = 'app-v1';
const RUNTIME_CACHE = 'runtime-v1';
// Assets to cache on install
const PRECACHE_ASSETS = [
'/',
'/offline.html',
'/styles/main.css',
'/scripts/main.js',
];
// Install: Cache critical assets
self.addEventListener('install', (event: ExtendableEvent) => {
event.waitUntil(
caches.open(CACHE_NAME)
.then(cache => cache.addAll(PRECACHE_ASSETS))
.then(() => self.skipWaiting())
);
});
// Activate: Clean old caches
self.addEventListener('activate', (event: ExtendableEvent) => {
event.waitUntil(
caches.keys()
.then(keys => Promise.all(
keys
.filter(key => key !== CACHE_NAME && key !== RUNTIME_CACHE)
.map(key => caches.delete(key))
))
.then(() => self.clients.claim())
);
});
// Fetch: Network first, fallback to cache
self.addEventListener('fetch', (event: FetchEvent) => {
const { request } = event;
// API requests: Network first
if (request.url.includes('/api/')) {
event.respondWith(
fetch(request)
.then(response => {
const clone = response.clone();
caches.open(RUNTIME_CACHE)
.then(cache => cache.put(request, clone));
return response;
})
.catch(() => caches.match(request))
);
return;
}
// Static assets: Cache first
event.respondWith(
caches.match(request)
.then(cached => cached || fetch(request))
.catch(() => caches.match('/offline.html'))
);
});
CDN Caching
Edge Caching with Vercel:
// pages/api/data.ts
import type { NextApiRequest, NextApiResponse } from 'next';
export default async function handler(
req: NextApiRequest,
res: NextApiResponse
) {
// Set cache headers for CDN
res.setHeader(
'Cache-Control',
'public, s-maxage=3600, stale-while-revalidate=86400'
);
const data = await fetchData();
res.status(200).json(data);
}
// Result: 95% of requests served from edge, latency reduced from 200ms to 20ms
API Response Caching
React Query for Smart Caching:
// lib/api-client.ts
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
export function useUser(userId: string) {
return useQuery({
queryKey: ['user', userId],
queryFn: () => fetch(`/api/users/${userId}`).then(r => r.json()),
staleTime: 5 * 60 * 1000, // 5 minutes
cacheTime: 10 * 60 * 1000, // 10 minutes
retry: 3,
refetchOnWindowFocus: false,
});
}
export function useUpdateUser() {
const queryClient = useQueryClient();
return useMutation({
mutationFn: (data: UpdateUserData) =>
fetch('/api/users', {
method: 'PUT',
body: JSON.stringify(data),
}),
onSuccess: (_, variables) => {
// Invalidate and refetch
queryClient.invalidateQueries(['user', variables.id]);
},
});
}
// Usage
function UserProfile({ userId }: { userId: string }) {
const { data, isLoading } = useUser(userId);
if (isLoading) return <Skeleton />;
return <Profile user={data} />;
}
Backend Optimization
Database Query Optimization
Index Strategy:
-- BEFORE: Full table scan (5000ms for 1M rows)
SELECT * FROM orders
WHERE user_id = 123
AND status = 'pending'
ORDER BY created_at DESC;
-- AFTER: Composite index (15ms)
CREATE INDEX idx_orders_user_status_date
ON orders(user_id, status, created_at DESC);
-- Query with index hint
SELECT * FROM orders
WHERE user_id = 123
AND status = 'pending'
ORDER BY created_at DESC
LIMIT 20;
-- Result: 99.7% faster queries
N+1 Query Prevention:
// BEFORE: N+1 queries (1001 queries for 1000 posts)
async function getPosts() {
const posts = await db.post.findMany();
for (const post of posts) {
post.author = await db.user.findUnique({ where: { id: post.authorId } });
}
return posts;
}
// AFTER: Single query with join (1 query)
async function getPosts() {
return await db.post.findMany({
include: {
author: true,
},
});
}
// Or use DataLoader for batch loading
import DataLoader from 'dataloader';
const userLoader = new DataLoader(async (userIds: readonly string[]) => {
const users = await db.user.findMany({
where: { id: { in: [...userIds] } },
});
return userIds.map(id => users.find(u => u.id === id));
});
// Result: Query time reduced from 10s to 50ms
API Response Compression
Gzip and Brotli Compression:
// middleware/compression.ts
import compression from 'compression';
import { NextApiRequest, NextApiResponse } from 'next';
export function compressionMiddleware(
req: NextApiRequest,
res: NextApiResponse,
next: () => void
) {
compression({
level: 6, // Balance between speed and compression
threshold: 1024, // Only compress responses > 1KB
filter: (req, res) => {
if (req.headers['x-no-compression']) return false;
return compression.filter(req, res);
},
})(req, res, next);
}
// Result: Response size reduced by 70% (500KB -> 150KB)
Connection Pooling
Database Connection Pool:
// lib/db-pool.ts
import { Pool } from 'pg';
const pool = new Pool({
host: process.env.DB_HOST,
database: process.env.DB_NAME,
user: process.env.DB_USER,
password: process.env.DB_PASSWORD,
// Connection pool settings
max: 20, // Maximum connections
min: 5, // Minimum idle connections
idleTimeoutMillis: 30000, // Close idle connections after 30s
connectionTimeoutMillis: 2000, // Timeout for acquiring connection
});
// Graceful shutdown
process.on('SIGTERM', async () => {
await pool.end();
process.exit(0);
});
export async function query<T>(sql: string, params?: any[]): Promise<T[]> {
const client = await pool.connect();
try {
const result = await client.query(sql, params);
return result.rows;
} finally {
client.release(); // Return connection to pool
}
}
// Result: Reduced connection overhead from 50ms to 2ms per query
Horizontal Scaling
Load Balancing Strategy:
// lib/load-balancer.ts
export class RoundRobinLoadBalancer {
private currentIndex = 0;
constructor(private servers: string[]) {}
getNextServer(): string {
const server = this.servers[this.currentIndex];
this.currentIndex = (this.currentIndex + 1) % this.servers.length;
return server;
}
}
// Usage
const loadBalancer = new RoundRobinLoadBalancer([
'https://api-1.example.com',
'https://api-2.example.com',
'https://api-3.example.com',
]);
export async function fetchWithLoadBalancing(endpoint: string) {
const server = loadBalancer.getNextServer();
return fetch(`${server}${endpoint}`);
}
Network Optimization
HTTP/2 and HTTP/3
Server Push and Multiplexing:
// next.config.js
module.exports = {
async headers() {
return [
{
source: '/:path*',
headers: [
{
key: 'Link',
value: [
'</styles/critical.css>; rel=preload; as=style',
'</scripts/main.js>; rel=preload; as=script',
'</fonts/inter.woff2>; rel=preload; as=font; crossorigin',
].join(', '),
},
],
},
];
},
};
Resource Hints
Preload, Prefetch, Preconnect:
// components/Head.tsx
import Head from 'next/head';
export function OptimizedHead() {
return (
<Head>
{/* Preconnect to critical origins */}
<link rel="preconnect" href="https://api.example.com" />
<link rel="preconnect" href="https://cdn.example.com" />
<link rel="dns-prefetch" href="https://analytics.example.com" />
{/* Preload critical resources */}
<link
rel="preload"
href="/fonts/inter-var.woff2"
as="font"
type="font/woff2"
crossOrigin="anonymous"
/>
{/* Prefetch next page resources */}
<link rel="prefetch" href="/dashboard" />
<link rel="prefetch" href="/api/user-data" />
{/* Preload hero image */}
<link
rel="preload"
as="image"
href="/hero.webp"
imageSrcSet="/hero-mobile.webp 640w, /hero.webp 1280w"
/>
</Head>
);
}
// Result: Reduced latency for critical resources by 200-500ms
Critical CSS
Inline Critical CSS:
// pages/_document.tsx
import Document, { Html, Head, Main, NextScript } from 'next/document';
class MyDocument extends Document {
render() {
return (
<Html>
<Head>
{/* Inline critical CSS */}
<style dangerouslySetInnerHTML={{
__html: `
body { margin: 0; font-family: -apple-system, sans-serif; }
.above-fold { min-height: 100vh; }
.hero { background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); }
`
}} />
{/* Async load full stylesheet */}
<link
rel="stylesheet"
href="/styles/main.css"
media="print"
onLoad="this.media='all'"
/>
</Head>
<body>
<Main />
<NextScript />
</body>
</Html>
);
}
}
export default MyDocument;
React/Next.js Specific
React.memo and useMemo
Prevent Unnecessary Re-renders:
// BEFORE: Re-renders on every parent update
function ExpensiveList({ items, onSelect }) {
console.log('Rendering list with', items.length, 'items');
return (
<ul>
{items.map(item => (
<li key={item.id} onClick={() => onSelect(item)}>
{item.name}
</li>
))}
</ul>
);
}
// AFTER: Only re-renders when items change
import { memo, useMemo, useCallback } from 'react';
const ExpensiveList = memo(function ExpensiveList({ items, onSelect }) {
console.log('Rendering list with', items.length, 'items');
const sortedItems = useMemo(
() => [...items].sort((a, b) => a.name.localeCompare(b.name)),
[items]
);
return (
<ul>
{sortedItems.map(item => (
<ListItem key={item.id} item={item} onSelect={onSelect} />
))}
</ul>
);
}, (prevProps, nextProps) => {
return prevProps.items === nextProps.items;
});
const ListItem = memo(function ListItem({ item, onSelect }) {
return (
<li onClick={() => onSelect(item)}>
{item.name}
</li>
);
});
// Parent component
function Parent() {
const [items, setItems] = useState([]);
const handleSelect = useCallback((item) => {
console.log('Selected:', item);
}, []);
return <ExpensiveList items={items} onSelect={handleSelect} />;
}
// Result: Reduced re-renders by 90%, improved interaction responsiveness
Virtual Scrolling
Handle Large Lists Efficiently:
// components/VirtualList.tsx
import { useVirtualizer } from '@tanstack/react-virtual';
import { useRef } from 'react';
interface VirtualListProps<T> {
items: T[];
renderItem: (item: T, index: number) => React.ReactNode;
estimateSize?: number;
}
export function VirtualList<T>({
items,
renderItem,
estimateSize = 50
}: VirtualListProps<T>) {
const parentRef = useRef<HTMLDivElement>(null);
const virtualizer = useVirtualizer({
count: items.length,
getScrollElement: () => parentRef.current,
estimateSize: () => estimateSize,
overscan: 5, // Render 5 extra items above/below viewport
});
return (
<div
ref={parentRef}
style={{
height: '600px',
overflow: 'auto',
}}
>
<div
style={{
height: `${virtualizer.getTotalSize()}px`,
width: '100%',
position: 'relative',
}}
>
{virtualizer.getVirtualItems().map((virtualItem) => (
<div
key={virtualItem.key}
style={{
position: 'absolute',
top: 0,
left: 0,
width: '100%',
height: `${virtualItem.size}px`,
transform: `translateY(${virtualItem.start}px)`,
}}
>
{renderItem(items[virtualItem.index], virtualItem.index)}
</div>
))}
</div>
</div>
);
}
// Usage: Render 10,000 items smoothly
function App() {
const items = Array.from({ length: 10000 }, (_, i) => ({
id: i,
name: `Item ${i}`,
}));
return (
<VirtualList
items={items}
renderItem={(item) => (
<div style={{ padding: '10px', borderBottom: '1px solid #eee' }}>
{item.name}
</div>
)}
/>
);
}
// Result: 10,000 items rendered at 60fps instead of freezing browser
Dynamic Imports
Split Heavy Components:
// BEFORE: Chart library bundled in main (300KB)
import { Chart } from 'react-chartjs-2';
function Dashboard() {
return (
<div>
<Chart data={data} />
</div>
);
}
// AFTER: Load chart only when needed
import dynamic from 'next/dynamic';
const Chart = dynamic(() => import('react-chartjs-2').then(mod => mod.Chart), {
loading: () => <ChartSkeleton />,
ssr: false, // Disable SSR for this component
});
function Dashboard() {
const [showChart, setShowChart] = useState(false);
return (
<div>
<button onClick={() => setShowChart(true)}>Show Chart</button>
{showChart && <Chart data={data} />}
</div>
);
}
// Result: Initial bundle 300KB smaller, TTI improved by 1.2s
Anti-Patterns
Avoid these common performance mistakes:
Premature Optimization
// DON'T: Optimize before measuring const memoizedValue = useMemo(() => value * 2, [value]); // Unnecessary // DO: Measure first, optimize if needed const value = props.value * 2; // Simple calculation, no memo neededOver-fetching Data
// DON'T: Request unnecessary fields const user = await fetch('/api/users/123'); // Returns 50 fields // DO: Request only what you need const user = await fetch('/api/users/123?fields=name,email');Blocking Main Thread
// DON'T: Heavy computation on main thread const result = heavyComputation(largeArray); // Blocks UI // DO: Use Web Workers const worker = new Worker('/worker.js'); worker.postMessage(largeArray); worker.onmessage = (e) => setResult(e.data);Unnecessary State Updates
// DON'T: Update state in render function Component() { const [count, setCount] = useState(0); setCount(1); // Causes infinite loop } // DO: Update in effects or callbacks function Component() { const [count, setCount] = useState(0); useEffect(() => setCount(1), []); }Ignoring Bundle Size
// DON'T: Import entire library import _ from 'lodash'; // 70KB // DO: Import only needed functions import debounce from 'lodash/debounce'; // 5KB
Examples
Example 1: Optimizing Initial Load
Scenario: Landing page takes 5.2s to load, LCP is 4.8s
// BEFORE: All resources loaded synchronously
import Hero from './Hero';
import Features from './Features';
import Testimonials from './Testimonials';
import Footer from './Footer';
import './styles/main.css';
import './styles/animations.css';
function LandingPage() {
return (
<>
<Hero />
<Features />
<Testimonials />
<Footer />
</>
);
}
// AFTER: Optimized loading strategy
import dynamic from 'next/dynamic';
import { Suspense } from 'react';
// Critical above-the-fold content
import Hero from './Hero';
// Lazy load below-the-fold components
const Features = dynamic(() => import('./Features'));
const Testimonials = dynamic(() => import('./Testimonials'));
const Footer = dynamic(() => import('./Footer'));
function LandingPage() {
return (
<>
{/* Above-the-fold: Load immediately */}
<Hero />
{/* Below-the-fold: Lazy load */}
<Suspense fallback={<div style={{ height: '400px' }} />}>
<Features />
</Suspense>
<Suspense fallback={<div style={{ height: '600px' }} />}>
<Testimonials />
</Suspense>
<Suspense fallback={<div style={{ height: '200px' }} />}>
<Footer />
</Suspense>
</>
);
}
// Split CSS
// main.css: Critical styles only (inline in <head>)
// animations.css: Load async
// Result:
// - Initial bundle: 450KB -> 120KB (73% reduction)
// - LCP: 4.8s -> 1.9s (60% improvement)
// - TTI: 5.2s -> 2.4s (54% improvement)
Example 2: Optimizing Data Fetching
Scenario: Dashboard makes 50+ API calls, takes 3s to load
// BEFORE: Sequential API calls
async function Dashboard() {
const user = await fetch('/api/user').then(r => r.json());
const posts = await fetch('/api/posts').then(r => r.json());
const comments = await fetch('/api/comments').then(r => r.json());
const analytics = await fetch('/api/analytics').then(r => r.json());
// Each request waits for previous: 3000ms total
}
// AFTER: Parallel fetching with React Query
import { useQueries } from '@tanstack/react-query';
function Dashboard() {
const results = useQueries({
queries: [
{ queryKey: ['user'], queryFn: () => fetch('/api/user').then(r => r.json()) },
{ queryKey: ['posts'], queryFn: () => fetch('/api/posts').then(r => r.json()) },
{ queryKey: ['comments'], queryFn: () => fetch('/api/comments').then(r => r.json()) },
{ queryKey: ['analytics'], queryFn: () => fetch('/api/analytics').then(r => r.json()) },
],
});
const [userQuery, postsQuery, commentsQuery, analyticsQuery] = results;
if (results.some(q => q.isLoading)) {
return <DashboardSkeleton />;
}
return (
<div>
<UserHeader user={userQuery.data} />
<PostsList posts={postsQuery.data} />
<CommentsPanel comments={commentsQuery.data} />
<AnalyticsChart data={analyticsQuery.data} />
</div>
);
}
// Even better: Server-side aggregation
// pages/api/dashboard.ts
export default async function handler(req, res) {
const [user, posts, comments, analytics] = await Promise.all([
db.user.findUnique({ where: { id: req.userId } }),
db.post.findMany({ where: { userId: req.userId } }),
db.comment.findMany({ where: { userId: req.userId } }),
getAnalytics(req.userId),
]);
res.json({ user, posts, comments, analytics });
}
// Result:
// - API calls: 50 -> 1 (98% reduction)
// - Load time: 3000ms -> 400ms (87% improvement)
// - Network traffic: 2MB -> 200KB (90% reduction)
Example 3: Image Gallery Optimization
Scenario: Image gallery with 100 high-res images crashes mobile browsers
// BEFORE: All images loaded at once
function Gallery({ images }: { images: string[] }) {
return (
<div className="grid">
{images.map(url => (
<img key={url} src={url} alt="" />
))}
</div>
);
}
// AFTER: Virtual grid with lazy loading
import { useVirtualizer } from '@tanstack/react-virtual';
import { useRef } from 'react';
import Image from 'next/image';
function Gallery({ images }: { images: string[] }) {
const parentRef = useRef<HTMLDivElement>(null);
const columnCount = 3;
const rowCount = Math.ceil(images.length / columnCount);
const rowVirtualizer = useVirtualizer({
count: rowCount,
getScrollElement: () => parentRef.current,
estimateSize: () => 300, // Row height
overscan: 2,
});
return (
<div
ref={parentRef}
style={{ height: '100vh', overflow: 'auto' }}
>
<div style={{ height: `${rowVirtualizer.getTotalSize()}px` }}>
{rowVirtualizer.getVirtualItems().map((virtualRow) => {
const startIndex = virtualRow.index * columnCount;
const rowImages = images.slice(startIndex, startIndex + columnCount);
return (
<div
key={virtualRow.key}
style={{
position: 'absolute',
top: 0,
left: 0,
width: '100%',
height: `${virtualRow.size}px`,
transform: `translateY(${virtualRow.start}px)`,
display: 'grid',
gridTemplateColumns: 'repeat(3, 1fr)',
gap: '10px',
}}
>
{rowImages.map((url, i) => (
<Image
key={startIndex + i}
src={url}
alt=""
width={400}
height={300}
loading="lazy"
placeholder="blur"
blurDataURL={generateBlurPlaceholder(url)}
sizes="(max-width: 768px) 100vw, 33vw"
/>
))}
</div>
);
})}
</div>
</div>
);
}
// Result:
// - Memory usage: 2GB -> 200MB (90% reduction)
// - Images loaded: 100 -> 15 (only visible ones)
// - Scroll FPS: 15fps -> 60fps (smooth scrolling)
// - Mobile: No crashes, works on low-end devices
Example 4: State Management Optimization
Scenario: Complex form with 50+ fields re-renders entire component on every keystroke
// BEFORE: Single state object causes full re-renders
function ComplexForm() {
const [formData, setFormData] = useState({
firstName: '',
lastName: '',
email: '',
// ... 47 more fields
});
const handleChange = (field: string, value: string) => {
setFormData(prev => ({ ...prev, [field]: value }));
// Entire form re-renders on every keystroke
};
return (
<form>
<Input value={formData.firstName} onChange={e => handleChange('firstName', e.target.value)} />
<Input value={formData.lastName} onChange={e => handleChange('lastName', e.target.value)} />
{/* 48 more inputs... */}
</form>
);
}
// AFTER: Isolated state with React Hook Form
import { useForm, Controller } from 'react-hook-form';
function ComplexForm() {
const { control, handleSubmit } = useForm({
defaultValues: {
firstName: '',
lastName: '',
email: '',
// ... 47 more fields
},
});
const onSubmit = (data) => {
console.log(data);
};
return (
<form onSubmit={handleSubmit(onSubmit)}>
<Controller
name="firstName"
control={control}
render={({ field }) => (
<Input {...field} />
// Only this input re-renders on change
)}
/>
<Controller
name="lastName"
control={control}
render={({ field }) => <Input {...field} />}
/>
{/* 48 more controlled inputs... */}
</form>
);
}
// Result:
// - Re-renders per keystroke: 50 -> 1 (98% reduction)
// - Input lag: 200ms -> 0ms (instant feedback)
// - CPU usage: 80% -> 5% while typing
Example 5: API Response Optimization
Scenario: Product list API returns 500KB of data, most unused
// BEFORE: Over-fetching data
// GET /api/products returns full product objects
[
{
id: 1,
name: "Product A",
description: "Long description...",
fullSpecifications: { /* 100 fields */ },
reviews: [ /* 50 reviews */ ],
relatedProducts: [ /* 10 products */ ],
images: [ /* 20 images */ ],
// ... lots more data
},
// ... 99 more products
]
// 500KB response for simple product list
// AFTER: GraphQL with field selection
import { gql, useQuery } from '@apollo/client';
const GET_PRODUCTS = gql`
query GetProducts($limit: Int!) {
products(limit: $limit) {
id
name
price
thumbnail
}
}
`;
function ProductList() {
const { data, loading } = useQuery(GET_PRODUCTS, {
variables: { limit: 100 },
});
if (loading) return <Skeleton />;
return (
<div className="grid">
{data.products.map(product => (
<ProductCard key={product.id} product={product} />
))}
</div>
);
}
// Or with REST: Field filtering
// GET /api/products?fields=id,name,price,thumbnail
// Result:
// - Response size: 500KB -> 25KB (95% reduction)
// - Load time: 2000ms -> 150ms (92.5% improvement)
// - Bandwidth saved: 475KB per request
// - Mobile data usage: Significantly reduced
Example 6: Database Query Optimization
Scenario: User dashboard query takes 8 seconds with N+1 problem
// BEFORE: N+1 queries
async function getUserDashboard(userId: string) {
// 1 query to get user
const user = await db.user.findUnique({
where: { id: userId },
});
// 1 query to get user's posts
const posts = await db.post.findMany({
where: { authorId: userId },
});
// N queries to get each post's comments (100 posts = 100 queries)
for (const post of posts) {
post.comments = await db.comment.findMany({
where: { postId: post.id },
});
// N queries to get each comment's author (500 comments = 500 queries)
for (const comment of post.comments) {
comment.author = await db.user.findUnique({
where: { id: comment.authorId },
});
}
}
return { user, posts };
}
// Total: 1 + 1 + 100 + 500 = 602 queries (8000ms)
// AFTER: Optimized with joins and includes
async function getUserDashboard(userId: string) {
return await db.user.findUnique({
where: { id: userId },
include: {
posts: {
include: {
comments: {
include: {
author: {
select: {
id: true,
name: true,
avatar: true,
},
},
},
},
},
},
},
});
}
// Total: 1 query (50ms)
// Result:
// - Queries: 602 -> 1 (99.8% reduction)
// - Query time: 8000ms -> 50ms (99.4% improvement)
// - Database load: Drastically reduced
// - Scalability: Can handle 20x more concurrent users
Remember: Performance optimization is an ongoing process. Continuously monitor your metrics, profile your application, and iterate on improvements. The goal is not perfection but providing users with a fast, responsive experience that meets your performance budgets.