React 中hooks之useDeferredValue用法总结

目录

  1. 概述
  2. 基本用法
  3. 与防抖节流的区别
  4. 使用场景
  5. 区分过时内容
  6. 最佳实践

概述

什么是 useDeferredValue?

useDeferredValue 是 React 18 引入的新 Hook,用于延迟更新某个不那么重要的部分。它接收一个值并返回该值的新副本,新副本会延迟更新。这种延迟是有益的,让紧急更新(如用户输入)优先于不紧急的更新(如渲染搜索结果列表)。

主要特点

  • 自动处理延迟
  • 与 Suspense 集成
  • 不会像防抖和useTransition那样丢弃中间值
  • 可以区分过时/最新内容

基本用法

1. 基本语法

typescript 复制代码
import { useDeferredValue } from 'react';

function SearchResults() {
  const [query, setQuery] = useState('');
  const deferredQuery = useDeferredValue(query);
  
  return (
    <div>
      <input value={query} onChange={e => setQuery(e.target.value)} />
      <SlowList query={deferredQuery} /> {/* 使用延迟值进行渲染 */}
    </div>
  );
}

2. 区分过时内容示例

typescript 复制代码
function SearchResults() {
  const [query, setQuery] = useState('');
  const deferredQuery = useDeferredValue(query);
  const isStale = query !== deferredQuery; // 判断内容是否过时
  
  return (
    <div>
      <input value={query} onChange={e => setQuery(e.target.value)} />
      <div style={{ 
        opacity: isStale ? 0.8 : 1,
        transition: 'opacity 0.2s ease'
      }}>
        <SlowList query={deferredQuery} />
      </div>
      {isStale && <div>Loading new results...</div>}
    </div>
  );
}

与防抖节流的区别

1. 防抖 (Debounce)

typescript 复制代码
// 防抖示例
function SearchWithDebounce() {
  const [query, setQuery] = useState('');
  
  const debouncedSearch = useCallback(
    debounce((value) => {
      // 执行搜索
      performSearch(value);
    }, 500),
    []
  );
  
  // 会丢弃中间值,只处理最后一次输入
  return (
    <input 
      onChange={e => {
        setQuery(e.target.value);
        debouncedSearch(e.target.value);
      }} 
    />
  );
}

2. 节流 (Throttle)

typescript 复制代码
// 节流示例
function SearchWithThrottle() {
  const [query, setQuery] = useState('');
  
  const throttledSearch = useCallback(
    throttle((value) => {
      performSearch(value);
    }, 100),
    []
  );
  
  // 固定时间间隔执行,可能会延迟响应
  return (
    <input 
      onChange={e => {
        setQuery(e.target.value);
        throttledSearch(e.target.value);
      }} 
    />
  );
}

3. useDeferredValue

typescript 复制代码
// useDeferredValue 示例
function SearchWithDeferred() {
  const [query, setQuery] = useState('');
  const deferredQuery = useDeferredValue(query);
  
  // React 会根据用户设备性能和当前 CPU 负载自动调整延迟
  // 不会丢弃任何值,而是以较低优先级处理它们
  return (
    <>
      <input onChange={e => setQuery(e.target.value)} />
      <SlowList query={deferredQuery} />
    </>
  );
}

使用场景

1. 大列表渲染

typescript 复制代码
function VirtualizedList({ items }) {
  const [filter, setFilter] = useState('');
  const deferredFilter = useDeferredValue(filter);
  const isStale = filter !== deferredFilter;
  
  const filteredItems = useMemo(
    () => items.filter(item => item.includes(deferredFilter)),
    [deferredFilter, items]
  );
  
  return (
    <div>
      <input value={filter} onChange={e => setFilter(e.target.value)} />
      <div style={{ opacity: isStale ? 0.8 : 1 }}>
        {filteredItems.map(item => (
          <ListItem key={item} item={item} />
        ))}
      </div>
    </div>
  );
}

2. 实时预览

typescript 复制代码
function MarkdownEditor() {
  const [text, setText] = useState('');
  const deferredText = useDeferredValue(text);
  const isStale = text !== deferredText;
  
  return (
    <div className="editor">
      <textarea
        value={text}
        onChange={e => setText(e.target.value)}
      />
      <div className={`preview ${isStale ? 'stale' : ''}`}>
        <MarkdownPreview text={deferredText} />
      </div>
    </div>
  );
}

最佳实践

  1. 合理使用 useMemo
typescript 复制代码
function SlowList({ text }) {
  const deferredText = useDeferredValue(text);
  const isStale = text !== deferredText;
  
  // 使用 useMemo 避免不必要的重新计算
  const items = useMemo(
    () => computeExpensiveList(deferredText),
    [deferredText]
  );
  
  return (
    <div style={{ opacity: isStale ? 0.8 : 1 }}>
      {items.map(item => (
        <ListItem key={item.id} item={item} />
      ))}
    </div>
  );
}
  1. 优雅降级处理
typescript 复制代码
function SearchResults({ query }) {
  const deferredQuery = useDeferredValue(query);
  const isStale = query !== deferredQuery;
  
  return (
    <div>
      {isStale && (
        <div className="stale-indicator">
          Showing results for "{deferredQuery}"
          <br />
          Updating results for "{query}"...
        </div>
      )}
      <ResultsList query={deferredQuery} />
    </div>
  );
}

总结

  1. useDeferredValue vs 防抖/节流:

    • useDeferredValue 不会丢弃更新
    • 自动适应用户设备性能
    • 与 React 并发特性集成
    • 提供过时状态标识
  2. 适用场景:

    • 大数据列表渲染
    • 实时预览功能
    • 复杂图表更新
    • 搜索建议
  3. 最佳实践:

    • 配合 useMemo 使用
    • 提供加载状态反馈
    • 合理处理过时内容
    • 注意性能优化
相关推荐
killerbasd3 小时前
牧苏苏传 我不装了 4/7
前端·javascript·vue.js
吴声子夜歌3 小时前
ES6——二进制数组详解
前端·ecmascript·es6
码事漫谈4 小时前
手把手带你部署本地模型,让你Token自由(小白专属)
前端·后端
ZC跨境爬虫4 小时前
【爬虫实战对比】Requests vs Scrapy 笔趣阁小说爬虫,从单线程到高效并发的全方位升级
前端·爬虫·scrapy·html
爱上好庆祝4 小时前
svg图片
前端·css·学习·html·css3
橘子编程4 小时前
JavaScript与TypeScript终极指南
javascript·ubuntu·typescript
王夏奇4 小时前
python中的__all__ 具体用法
java·前端·python
叫我一声阿雷吧5 小时前
JS 入门通关手册(45):浏览器渲染原理与重绘重排(性能优化核心,面试必考
javascript·前端面试·前端性能优化·浏览器渲染·浏览器渲染原理,重排重绘·reflow·repaint
大家的林语冰5 小时前
《前端周刊》尤大开源 Vite+ 全家桶,前端工业革命启动;尤大爆料 Void 云服务新产品,Vite 进军全栈开发;ECMA 源码映射规范......
前端·javascript·vue.js
jiayong235 小时前
第 8 课:开始引入组合式函数
前端·javascript·学习