React 中hooks之useTransition使用总结
目录
- 概述
- 基本用法
- 使用场景
- 最佳实践
- 注意事项
概述
什么是 useTransition?
useTransition 是 React 18 引入的新 Hook,用于标记非紧急的状态更新。它允许组件在状态转换期间保持响应,通过将某些更新标记为"过渡"来推迟它们的渲染。
主要特点
- 保持 UI 响应性
- 区分紧急和非紧急更新
- 提供加载状态指示器
- 不会阻塞用户交互
基本用法
1. 基本语法
import { useTransition } from 'react';function MyComponent() {const [isPending, startTransition] = useTransition();const [count, setCount] = useState(0);const handleClick = () => {startTransition(() => {setCount(c => c + 1); // 这个更新被标记为过渡,当某次更新造成页面阻塞时,用户点击其他组件操作,此时会降低此次更新的优先级,不阻塞页面渲染先更新优先级高的操作,startTransition中只能写同步的代码,异步代码会打断低优先级,比如不能使用setTimeout});};return (<div>{isPending && <Spinner />}<button onClick={handleClick}>Increment</button><p>Count: {count}</p></div>);
}//isPending为true是表示当前渲染阻塞避免页面出现卡顿现象先显示loading状态组件,直到isPending为false时将展示渲染好的组件
2. 带加载状态的示例
function SearchResults() {const [query, setQuery] = useState('');const [results, setResults] = useState([]);const [isPending, startTransition] = useTransition();const handleSearch = (e: React.ChangeEvent<HTMLInputElement>) => {// 立即更新输入值(紧急更新)setQuery(e.target.value);// 将搜索结果更新标记为过渡(非紧急更新)startTransition(() => {// 模拟搜索操作const searchResults = performSearch(e.target.value);setResults(searchResults);});};return (<div><input value={query} onChange={handleSearch} />{isPending ? (<div>Loading...</div>) : (<ul>{results.map(result => (<li key={result.id}>{result.title}</li>))}</ul>)}</div>);
}
使用场景
1. 大量数据渲染
function DataGrid() {const [data, setData] = useState([]);const [isPending, startTransition] = useTransition();const [filter, setFilter] = useState('');const handleFilterChange = (newFilter: string) => {setFilter(newFilter); // 立即更新过滤条件startTransition(() => {// 延迟大量数据的过滤和渲染const filteredData = processLargeDataSet(newFilter);setData(filteredData);});};return (<div><input value={filter} onChange={e => handleFilterChange(e.target.value)} />{isPending ? (<LoadingGrid />) : (<VirtualizedGrid data={data} />)}</div>);
}
2. 路由切换
function App() {const [isPending, startTransition] = useTransition();const [currentPage, setCurrentPage] = useState('home');const navigate = (page: string) => {// 立即更新导航状态startTransition(() => {setCurrentPage(page);});};return (<div><Navigation onNavigate={navigate} />{isPending ? (<PageTransitionSpinner />) : (<Page name={currentPage} />)}</div>);
}
3. 表单验证
function ComplexForm() {const [formData, setFormData] = useState({});const [errors, setErrors] = useState({});const [isPending, startTransition] = useTransition();const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {const { name, value } = e.target;// 立即更新表单值setFormData(prev => ({...prev,[name]: value}));// 将复杂的验证逻辑标记为过渡startTransition(() => {const validationErrors = validateFormField(name, value);setErrors(prev => ({...prev,[name]: validationErrors}));});};return (<form><input name="email" onChange={handleChange} value={formData.email || ''} />{isPending ? (<ValidatingIndicator />) : (errors.email && <ErrorMessage error={errors.email} />)}</form>);
}
最佳实践
- 区分更新优先级
function UserProfile() {const [isPending, startTransition] = useTransition();const handleInput = (e: React.ChangeEvent<HTMLInputElement>) => {// 高优先级:直接更新输入值setInputValue(e.target.value);// 低优先级:更新预览startTransition(() => {setPreviewData(generatePreview(e.target.value));});};
}
- 合理使用 isPending
function LoadingStates() {const [isPending, startTransition] = useTransition();return (<div>{/* 使用骨架屏而不是简单的加载指示器 */}{isPending ? (<Skeleton />) : (<Content />)}</div>);
}
注意事项
- 不要在 transition 中包含紧急更新
// ❌ 错误示例
startTransition(() => {setInputValue(e.target.value); // 这应该是紧急更新
});// ✅ 正确示例
setInputValue(e.target.value); // 紧急更新
startTransition(() => {setSearchResults(search(e.target.value)); // 非紧急更新
});
- 避免不必要的 transition
// ❌ 不需要 transition
startTransition(() => {setCount(count + 1); // 简单的状态更新不需要 transition
});// ✅ 适合使用 transition
startTransition(() => {setFilteredItems(items.filter(complexFilter)); // 复杂计算
});
总结
-
useTransition 适用场景:
- 大量数据处理
- 复杂 UI 更新
- 后台计算
- 非阻塞渲染
-
主要优势:
- 提升用户体验
- 保持 UI 响应性
- 优化渲染性能
- 提供加载状态
-
使用建议:
- 合理区分更新优先级
- 适当处理加载状态
- 避免过度使用
- 配合其他性能优化手段
相关文章:
React 中hooks之useTransition使用总结
目录 概述基本用法使用场景最佳实践注意事项 概述 什么是 useTransition? useTransition 是 React 18 引入的新 Hook,用于标记非紧急的状态更新。它允许组件在状态转换期间保持响应,通过将某些更新标记为"过渡"来推迟它们的渲染。 主要特…...
怎样使用树莓派自己搭建一套ADS-B信号接收系统
0 我们知道,ADS-B全称广播式自动相关监视系统,其实就是飞机发出的广播信号,用明码来对外发送自己的位置、高度、速度、航向等信息,是公开信息。连续接收到一架飞机发出的ADS-B信息后,可以通过其坐标点来描绘出飞机的航…...
Chrome谷歌浏览器如何能恢复到之前的旧版本
升级了谷歌最新版不习惯,如何降级版本 未完待续。。 电脑中的Chrome谷歌浏览器升级到了最新版本,但是有种种的不适应,如何能恢复到之前的旧版本呢?我们来看看操作步骤,而且无需卸载重装。 怎么恢复Chrome 之前版本&a…...
路由器旁挂三层网络实现SDWAN互联(爱快SD-WAN)
近期因公司新办公区建设,原有的爱快路由器的SDWAN功能实现分支之间互联的服务还需要继续使用。在原有的小型网络中,使用的爱快路由器当作网关设备,所以使用较为简单,如下图所示。 现变更网络拓扑为三层网络架构,但原有的SDWAN分支…...
代码随想录算法训练营第五十五天 |108.冗余连接 109.冗余连接Ⅱ
108.冗余连接: 文章链接 题目链接:108.冗余连接 思路 首先分析题目,给定拥有n个节点和n条边的图,其中图是在原n个节点和n - 1条无环无向图中添加一条边得到的。要求是输出多出的边。(PS:可能会有多个答案…...
Unity补充 -- 协程相关
1.协程。 协程并不是线程。线程是主线程之外的另一条 代码按照逻辑执行通道。协程则是在代码在按照逻辑执行的同时,是否需要执行额外的语句块。 2.协程的作用。 在update执行的时候,是按照帧来进行刷新的,也是按照帧执行代码的。但是又不想…...
【第二十周】U-Net:用于生物图像分割的卷积神经网络
文章目录 摘要Abstract文章信息研究动机U-Net网络结构U-Net网络搭建数据增强损失函数转置卷积创新性与不足创新性:不足: 总结 摘要 U-Net(Convolutional Networks for Biomedical Image Segmentation)是一种用于图像分割的深度学…...
部署Metricbeat监测ES
官方参考文档 安装Metricbeat curl -L -O https://artifacts.elastic.co/downloads/beats/metricbeat/metricbeat-7.17.27-linux-x86_64.tar.gztar xzvf metricbeat-7.17.27-linux-x86_64.tar.gz设置 Metricbeat连接到 Elasticsearch 进入metricbeat目录配置metricbeat.yml …...
Pytorch|YOLO
🍨 本文为🔗365天深度学习训练营中的学习记录博客🍖 原作者:K同学啊 一、 前期准备 1. 设置GPU 如果设备上支持GPU就使用GPU,否则使用CPU import torch import torch.nn as nn import torchvision.transforms as transforms im…...
云计算与物联网技术的融合应用(在工业、农业、家居、医疗、环境、城市等整理较全)
摘要 为生产领域带来更加全面和深入的变革。通过云计算平台对物联网数据进行处理和分析,企业可以实现对生产过程的更加精细化的管理和控制。 1. 智能生产调度 通过云计算和物联网技术的融合应用,企业可以实现对生产线上各个环节的实时监控和数据分析。…...
基于python+Django+mysql鲜花水果销售商城网站系统设计与实现
博主介绍:黄菊华老师《Vue.js入门与商城开发实战》《微信小程序商城开发》图书作者,CSDN博客专家,在线教育专家,CSDN钻石讲师;专注大学生毕业设计教育、辅导。 所有项目都配有从入门到精通的基础知识视频课程ÿ…...
Golang:报错no required module provides package github.com/xx的解决方法
报错 问题重现可能的原因及解决方法1. 未初始化 Go 模块解决方法: 2. 没有添加依赖解决方法: 3. 网络问题解决方法: 4. 依赖版本问题解决方法: 5. 包未发布或路径拼写错误解决方法: 6. go mod tidy 未运行解决方法&…...
数据结构与算法(2):顺序表与链表
1.前言 哈喽大家好喔,今天博主继续进行数据结构的分享与学习,今天的主要内容是顺序表与链表,是最简单但又相当重要的数据结构,为以后的学习有重要的铺垫,希望大家一起交流学习,互相进步,让我们…...
华为OD机试E卷 --过滤组合字符串--24年OD统一考试(Java JS Python C C++)
文章目录 题目描述输入描述输出描述用例题目解析JS算法源码Java算法源码python算法源码c算法源码c++算法源码题目描述 数字 0、1、2、3、4、5、6、7、8、9 分别关联 a~z 26 个英文字母。 0 关联“a”"b”"c1 关联“d”"e”"f2 关联“g"“h”“i”3 关…...
QT跨平台应用程序开发框架(3)—— 信号和槽
目录 一,基本概念 二,connect函数使用 2.1 connect 2.2 Qt内置信号和槽 2.3 一些细节 三,自定义信号和槽 3.1 自定义槽函数 3.2 自定义信号 3.3 带参数的信号槽 四,信号和槽的意义 五,信号和槽断开连接 六&…...
从 0 开始实现一个 SpringBoot + Vue 项目
从 0 开始实现一个 SpringBoot Vue 项目 从 0 开始实现一个 SpringBoot Vue 项目 软件和工具创建 SpringBoot 后端项目创建 MySQL 数据库配置文件实现增删改查接口 Model 层mapper 层service 层controller 层测试 实现项目功能接口 代码测试 创建 Vue 前端 安装 Node.js配置…...
【无标题】微调是迁移学习吗?
是的,微调(Fine-Tuning)可以被视为一种迁移学习(Transfer Learning)的形式。迁移学习是一种机器学习方法,其核心思想是利用在一个任务上学到的知识来改进另一个相关任务的性能。微调正是通过在预训练模型的…...
虚幻基础1:hello world
能帮到你的话,就给个赞吧 😘 文章目录 hello world创建项目创建关卡创建蓝图将蓝图插入关卡中运行 hello world 本文引擎为5.5.1 创建项目 如图 创建后如图。 创建关卡 如图 创建蓝图 如图 选择actor 双击进入蓝图节点 选择事件图表 创…...
C链表的一些基础知识
一、链表的基本概念 链表是一种常见的线性数据结构,它由一系列节点组成,每个节点包含数据部分和指向下一个节点的指针(单链表情况)。通过指针将各个节点连接起来,与数组不同,链表在内存中的存储不是连续的…...
JDK长期支持版本(LTS)
https://blogs.oracle.com/java/post/the-arrival-of-java-23 jdk长期支持版本(LTS):JDK 8、11、17、21:...
图片格式转换效率革命:从繁琐流程到一键操作的技术突破
图片格式转换效率革命:从繁琐流程到一键操作的技术突破 【免费下载链接】Save-Image-as-Type Save Image as Type is an chrome extension which add Save as PNG / JPG / WebP to the context menu of image. 项目地址: https://gitcode.com/gh_mirrors/sa/Save-…...
基于springboot+vue高校物资信息采购系统hx0807
文章目录详细视频演示技术介绍功能介绍核心代码系统效果图源码获取详细视频演示 文章底部名片,获取项目的完整演示视频,免费解答技术疑问 技术介绍 开发语言:Java 框架:ssm JDK版本:JDK1.8 服务器:tomca…...
从DINO Score到LLaVA:拆解SPAA论文如何用“双考官”机制筛选高质量AI修图
从DINO Score到LLaVA:构建AI图像编辑的"双考官"质量评估体系 在AI图像编辑技术快速发展的今天,如何系统评估生成结果的质量已成为产品落地的关键瓶颈。传统方法往往依赖人工审核或单一指标,既难以规模化又无法全面捕捉图像修改的语…...
HagiCode 为什么选择 Hermes 作为综合 Agent 核心一
1. 哑铃图是什么? 哑铃图(Dumbbell Plot),有时也称为DNA图或杠铃图,是一种用于比较两个相关数据点的可视化图表。 它源于人们对更有效数据比较方式的持续探索。 在传统的时间序列比较中,我们通常使用两条折…...
Windows 10/11 上保姆级安装AdGuard Home,并配置为开机自启服务(附NSSM详细步骤)
Windows 系统深度集成 AdGuard Home:从零构建企业级 DNS 过滤服务 在数字生活高度渗透的今天,网络隐私保护已成为现代计算机用户的刚需。作为 Windows 平台用户,我们常常面临一个两难选择:要么忍受各类广告追踪和恶意域名的侵扰&…...
CKKS 同态加密数学基础推导嗡
背景 StreamJsonRpc 是微软官方维护的用于 .NET 和 TypeScript 的 JSON-RPC 通信库,以其强大的类型安全、自动代理生成和成熟的异常处理机制著称。在 HagiCode 项目中,为了通过 ACP (Agent Communication Protocol) 与外部 AI 工具(如 iflow …...
振动力学实战:如何用MATLAB模拟无阻尼多自由度系统的受迫振动(附完整代码)
振动力学实战:MATLAB频域分析全流程与工程避坑指南 当一座斜拉桥在特定风速下出现异常振动,或是精密仪器因环境微振动导致测量失准,背后往往隐藏着多自由度系统的动力学奥秘。作为机械与土木工程师,掌握无阻尼多自由度系统的频域分…...
避开ArduPilot地面无人平台调试大坑:ACRO模式下的转向速率设置详解
ArduPilot无人平台ACRO模式转向调参实战:从参数解析到竞技级手感优化 第一次在空地上测试ArduPilot无人车时,我满心期待它能像竞技级RC模型那样做出精准的漂移过弯。但现实是——转向要么迟钝得像在泥沼里打转,要么突然变得过于敏感导致车辆原…...
一个Ingress搞定前后端分离:实战配置将API请求转发后端,静态页面留给前端
一个Ingress搞定前后端分离:实战配置将API请求转发后端,静态页面留给前端 在前后端分离架构成为主流的今天,如何优雅地部署应用成了开发者必须面对的挑战。想象一下:用户访问你的网站时,浏览器应该加载React或Vue构建的…...
QCustomPlot个性化踩坑实录:从默认丑图表到定制化美图,我总结了这几点经验
QCustomPlot个性化踩坑实录:从默认丑图表到定制化美图,我总结了这几点经验 第一次用QCustomPlot做工业监控系统的数据可视化时,我被它默认的"工程师审美"震惊了——灰底黑线、刻板的网格、毫无层次感的配色。更崩溃的是,…...
