關於React 效能最佳化 2024 指南,很多開發者只停留在 API 呼叫層面。本文試圖從生產環境的角度,討論實際中會遇到的問題和解決方案。
基本原理
在這個基礎上,我們可以進一步最佳化:
javascript
const observer = new PerformanceObserver((list) => {
for (const entry of list.getEntries()) {
if (entry.entryType === 'largest-contentful-paint') {
reportMetric('LCP', entry.startTime)
}
if (entry.entryType === 'first-input') {
reportMetric('FID', entry.processingStart - entry.startTime)
}
}
})
observer.observe({ entryTypes: ['largest-contentful-paint', 'first-input'] })
這種模式在大型專案中非常實用,能顯著降低維護成本。
高階特性
實際專案中的用法會更復雜一些:
javascript
import { useRef, useEffect, useState } from 'react'
function useIntersectionObserver(options = {}) {
const [isVisible, setIsVisible] = useState(false)
const ref = useRef(null)
useEffect(() => {
const observer = new IntersectionObserver(([entry]) => {
setIsVisible(entry.isIntersecting)
}, { threshold: 0.1, ...options })
const el = ref.current
if (el) observer.observe(el)
return () => { if (el) observer.unobserve(el) }
}, [])
return [ref, isVisible]
}
通過這種方式,程式碼的可測試性和可擴充套件性都得到了提升。
專案實踐
以下是一個完整的示例:
javascript
import { useReducer, useCallback } from 'react'
const initialState = { items: [], filter: '', sort: 'date' }
function reducer(state, action) {
switch (action.type) {
case 'SET_ITEMS': return { ...state, items: action.payload }
case 'SET_FILTER': return { ...state, filter: action.payload }
case 'ADD_ITEM': return { ...state, items: [...state.items, action.payload] }
case 'REMOVE_ITEM': return { ...state, items: state.items.filter(i => i.id !== action.payload) }
default: throw new Error(`Unknown: ${action.type}`)
}
}
注意邊界條件處理,這在生產環境中至關重要。
最佳實踐
關鍵在於理解核心邏輯:
javascript
import { useState, useEffect, useCallback } from 'react'
function DataList({ endpoint, pageSize = 20 }) {
const [data, setData] = useState([])
const [page, setPage] = useState(1)
const [loading, setLoading] = useState(false)
const fetchData = useCallback(async () => {
setLoading(true)
try {
const res = await fetch(`${endpoint}?page=${page}&size=${pageSize}`)
setData(await res.json())
} finally { setLoading(false) }
}, [endpoint, page, pageSize])
useEffect(() => { fetchData() }, [fetchData])
return <div>{loading ? <Spinner /> : <List items={data} />}</div>
}
效能最佳化需要結合具體場景,不是所有情況都需要過度最佳化。
踩坑記錄
我們可以通過以下方式來改進:
javascript
const observer = new PerformanceObserver((list) => {
for (const entry of list.getEntries()) {
if (entry.entryType === 'largest-contentful-paint') {
reportMetric('LCP', entry.startTime)
}
if (entry.entryType === 'first-input') {
reportMetric('FID', entry.processingStart - entry.startTime)
}
}
})
observer.observe({ entryTypes: ['largest-contentful-paint', 'first-input'] })
這套方案已經在線上穩定運行了半年以上,經過了實際驗證。
小結
- 不要為了用新技術而用新技術
- 程式碼示例僅供參考,需根據業務場景調整
- React 效能最佳化 2024 指南不是銀彈,需要根據專案規模和技術棧選擇
- 理解底層原理比記住 API 更重要