React 第五十六节 Router 中useSubmit的使用详解及注意事项
前言
useSubmit 是 React Router v6.4+ 引入的强大钩子,用于以编程方式提交表单数据。
它提供了对表单提交过程的精细控制,特别适合需要自定义提交行为或非标准表单场景的应用。
一、useSubmit 核心用途
- 编程式表单提交:不依赖
<form>元素手动提交数据 - 自定义提交逻辑:在提交前/后执行额外操作
- 动态表单构建:根据应用状态生成提交数据
- 替代传统表单:在复杂UI中实现表单功能
二、useSubmit 基本用法
import { useSubmit } from 'react-router-dom';function CustomSubmitButton() {const submit = useSubmit();const handleSubmit = () => {// 构建表单数据const formData = new FormData();formData.append('username', 'john_doe');formData.append('password', 'secure123');// 提交数据submit(formData, {method: 'post',action: '/login',encType: 'application/x-www-form-urlencoded'});};return (<button onClick={handleSubmit}>登录</button>);
}
三、useSubmit 参数详解
3.1、submit 函数参数
```javascript
submit(data: FormData | URLSearchParams | { [key: string]: string } | null,options?: {method?: 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';action?: string;encType?: 'application/x-www-form-urlencoded' | 'multipart/form-data';replace?: boolean;}
)
```
3.2、选项说明:
method:HTTP方法(默认为"GET")
action:提交的目标URL(默认为当前URL)
encType:表单编码类型(默认为"application/x-www-form-urlencoded")
replace:是否替换历史记录而不是添加新条目(默认为false)
四、useSubmit 实际应用场景
4.1、自定义删除按钮
import { useSubmit } from 'react-router-dom';function DeleteButton({ itemId }) {const submit = useSubmit();const handleDelete = () => {if (window.confirm('确定要删除此项吗?')) {submit({ id: itemId }, { method: 'delete', action: `/items/${itemId}` });}};return (<button onClick={handleDelete}className="delete-btn"aria-label="删除项目">🗑️ 删除</button>);
}
4.2、动态搜索表单
import { useState, useEffect } from 'react';
import { useSubmit, useLocation } from 'react-router-dom';function SearchBar() {const submit = useSubmit();const location = useLocation();const [query, setQuery] = useState('');// 使用防抖自动提交搜索useEffect(() => {const timerId = setTimeout(() => {if (query.trim() !== '') {const formData = new FormData();formData.append('q', query);submit(formData, {method: 'get',action: '/search',replace: query === '' // 空查询时替换历史记录});}}, 300);return () => clearTimeout(timerId);}, [query, submit]);// 初始化时从URL读取查询参数useEffect(() => {const params = new URLSearchParams(location.search);setQuery(params.get('q') || '');}, [location.search]);return (<div className="search-container"><inputtype="text"value={query}onChange={(e) => setQuery(e.target.value)}placeholder="搜索..."className="search-input"/>{query && (<button onClick={() => setQuery('')}className="clear-btn">✕</button>)}</div>);
}
4.3、多步骤表单
import { useState } from 'react';
import { useSubmit } from 'react-router-dom';function MultiStepForm() {const submit = useSubmit();const [step, setStep] = useState(1);const [formData, setFormData] = useState({personal: {},address: {},preferences: {}});const handleNextStep = () => {if (step < 3) {setStep(step + 1);} else {// 最终提交submitForm();}};const handlePrevStep = () => {setStep(step - 1);};const updateFormData = (section, data) => {setFormData(prev => ({...prev,[section]: { ...prev[section], ...data }}));};const submitForm = () => {// 构建完整表单数据const finalData = {...formData.personal,...formData.address,...formData.preferences};submit(finalData, {method: 'post',action: '/register',encType: 'application/json'});};return (<div className="multi-step-form"><div className="progress-bar"><div className={`step ${step >= 1 ? 'active' : ''}`}>个人信息</div><div className={`step ${step >= 2 ? 'active' : ''}`}>地址信息</div><div className={`step ${step >= 3 ? 'active' : ''}`}>偏好设置</div></div><div className="form-content">{step === 1 && (<PersonalInfo data={formData.personal}onChange={data => updateFormData('personal', data)}/>)}{step === 2 && (<AddressInfo data={formData.address}onChange={data => updateFormData('address', data)}/>)}{step === 3 && (<Preferences data={formData.preferences}onChange={data => updateFormData('preferences', data)}/>)}</div><div className="form-navigation">{step > 1 && (<button onClick={handlePrevStep}>上一步</button>)}<button onClick={handleNextStep}>{step < 3 ? '下一步' : '提交注册'}</button></div></div>);
}
4.4、文件上传
import { useRef, useState } from 'react';
import { useSubmit } from 'react-router-dom';function FileUploader() {const submit = useSubmit();const fileInputRef = useRef(null);const [isUploading, setIsUploading] = useState(false);const [progress, setProgress] = useState(0);const handleFileChange = async (e) => {const file = e.target.files[0];if (!file) return;setIsUploading(true);try {// 创建FormData对象const formData = new FormData();formData.append('file', file);formData.append('description', '用户上传的文件');// 创建XMLHttpRequest以获取上传进度const xhr = new XMLHttpRequest();// 进度事件处理xhr.upload.addEventListener('progress', (event) => {if (event.lengthComputable) {const percent = Math.round((event.loaded / event.total) * 100);setProgress(percent);}});// 完成事件处理xhr.addEventListener('load', () => {setIsUploading(false);setProgress(0);alert('文件上传成功!');});// 错误处理xhr.addEventListener('error', () => {setIsUploading(false);alert('文件上传失败');});// 打开并发送请求xhr.open('POST', '/upload');xhr.send(formData);// 或者使用submit钩子(但无法获取进度)// submit(formData, {// method: 'post',// action: '/upload',// encType: 'multipart/form-data'// });} catch (error) {setIsUploading(false);console.error('上传失败:', error);}};return (<div className="file-uploader"><input type="file" ref={fileInputRef} onChange={handleFileChange}style={{ display: 'none' }} /><button onClick={() => fileInputRef.current.click()}disabled={isUploading}>选择文件</button>{isUploading && (<div className="upload-progress"><div className="progress-bar" style={{ width: `${progress}%` }}></div><span>{progress}%</span></div>)}</div>);
}
4.5、乐观更新与表单提交
import { useState } from 'react';
import { useSubmit } from 'react-router-dom';function TodoItem({ todo }) {const submit = useSubmit();const [isCompleted, setIsCompleted] = useState(todo.completed);const [isUpdating, setIsUpdating] = useState(false);const handleToggle = async () => {// 乐观更新:立即更新UIconst newCompleted = !isCompleted;setIsCompleted(newCompleted);setIsUpdating(true);try {// 准备表单数据const formData = new FormData();formData.append('id', todo.id);formData.append('completed', newCompleted.toString());// 提交更新submit(formData, {method: 'patch',action: `/todos/${todo.id}`,encType: 'application/x-www-form-urlencoded'});} catch (error) {// 出错时回滚状态setIsCompleted(!newCompleted);console.error('更新失败:', error);} finally {setIsUpdating(false);}};return (<li className={`todo-item ${isCompleted ? 'completed' : ''}`}><inputtype="checkbox"checked={isCompleted}onChange={handleToggle}disabled={isUpdating}/><span className="todo-text">{todo.text}</span>{isUpdating && <span className="updating-indicator">更新中...</span>}</li>);
}
五、useSubmit 与相关API对比

六、useSubmit 最佳实践
-
数据验证:提交前验证数据
-
用户反馈:提交时显示加载状态
-
错误处理:捕获并处理提交错误
-
编码类型:
a.简单数据:
application/x-www-form-urlencodedb.文件上传:
multipart/form-datac.JSON数据:
手动序列化并设置适当headers -
性能优化:大文件上传使用分块或流式上传
// 带验证的提交示例
function ValidatedForm() {const submit = useSubmit();const [email, setEmail] = useState('');const [error, setError] = useState('');const handleSubmit = () => {// 验证邮箱格式if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)) {setError('请输入有效的邮箱地址');return;}// 提交数据submit({ email }, { method: 'post', action: '/subscribe' });};return (<div className="subscribe-form"><inputtype="email"value={email}onChange={(e) => {setEmail(e.target.value);setError('');}}placeholder="输入您的邮箱"className={error ? 'error' : ''}/>{error && <div className="error-message">{error}</div>}<button onClick={handleSubmit}>订阅</button></div>);
}
七、useSubmit 注意事项
-
路由上下文:必须在路由组件中使用(在
<Router>上下文中) -
数据格式:
对象会被转换为
URLSearchParams文件上传必须使用
FormData -
GET请求:数据会作为查询参数添加到URL
-
重定向:提交后服务器应返回重定向响应
-
状态管理:提交不会自动管理加载状态,需自行实现
总结
useSubmit 是 React Router 中处理表单提交的高级工具,特别适合以下场景:
- 自定义表单逻辑:当需要超出标准表单的行为时
- 动态数据提交:根据应用状态生成提交数据
- 非表单元素提交:从按钮或其他UI元素触发提交
- 复杂交互流程:如多步骤表单、乐观更新
- 文件上传:处理二进制数据提交
通过合理使用 useSubmit,您可以构建更灵活、更强大的表单交互体验,同时保持与 React Router 数据路由模型的无缝集成。
相关文章:
React 第五十六节 Router 中useSubmit的使用详解及注意事项
前言 useSubmit 是 React Router v6.4 引入的强大钩子,用于以编程方式提交表单数据。 它提供了对表单提交过程的精细控制,特别适合需要自定义提交行为或非标准表单场景的应用。 一、useSubmit 核心用途 编程式表单提交:不依赖 <form>…...
华为云学堂-云原生开发者认证课程列表
华为云学堂-云原生认证 云原生开发者认证的前5个课程...
Vue.js 组件:深入理解与实践
Vue.js 组件:深入理解与实践 引言 随着前端技术的不断发展,Vue.js 作为一种流行的前端框架,因其简洁、易学、高效的特点受到越来越多开发者的青睐。在Vue.js中,组件是构建用户界面的基石。本文将深入探讨Vue.js组件的概念、特性、创建方式以及在实际开发中的应用,帮助读…...
什么是强化学习:设置奖励函数最为loss, 监督学习:标签准确率作为loss
什么是强化学习:设置奖励函数最为loss, 监督学习:标签准确率作为loss 什么是强化学习:在复杂环境中自主探索,适用于序列决策 最大优势: 通过试错探索发现最优策略,适应环境动态变化,擅长解决需要长期规划和序列决策的问题。典型案例: 游戏AI(如AlphaGo/AlphaZero):…...
理解网络协议
1.查看网络配置 : ipconfig 2. ip地址 : ipv4(4字节, 32bit), ipv6, 用来标识主机的网络地址 3.端口号(0~65535) : 用来标识主机上的某个进程, 1 ~ 1024 知名端口号, 如果是服务端的话需要提供一个特定的端口号, 客户端的话是随机分配一个端口号 4.协议 : 简单来说就是接收数据…...
placeholder不显示and模板字符串无效
一、placeholder"请输入"不显示请输入? input框里写了placeholder为什么不显示呢? 检查代码,input是否有初始值 在 Vue.js 中,v-model 是双向绑定的语法糖,它会动态更新输入框的 value。如果绑定的数据有初…...
在MyBatis中设计SQL返回布尔值(Boolean)有几种常见方法
方案一:使用COUNT查询存在性(推荐) <select id"checkUserExists" resultType"_boolean">SELECT COUNT(*) > 0 FROM users WHERE username #{username} </select> 说明: MySQL中COU…...
全球知名具身智能/AI机器人实验室介绍之AI FACTORY基于慕尼黑工业大学
全球知名具身智能/AI机器人实验室介绍之AI FACTORY基于慕尼黑工业大学 TUM AI FACTORY,即KI.FABRIK,是德国慕尼黑工业大学(TUM)在巴伐利亚州推出的一个旗舰项目,旨在打造未来工厂,将传统工厂转变为由人工智…...
DASCTF
[DASCTF X 0psu3十一月挑战赛|越艰巨越狂热]EzPenetration Tip:数据库里的邮箱key已更改为管理员密码,拿到后可直接登录 打开靶机,用Wappalyzer分析网站,可以看到管理系统是Wordpress,因此可以尝试用WPSSCAN扫描公开…...
钉钉 - 机器人消息推送(签名版)
前言 在日常生活中,我们可能会遇到某些异常发生后需要紧急通知到群里,让相关人员看到紧急处理的事件触发机制。 消息群我采用的是钉钉推送,本文介绍了如何用php 推送钉钉机器人消息。 源码封装 <?php /*** 钉钉通知 - 签名版*/ class …...
Redux 实践与中间件应用
Redux 异步处理的挑战 Redux 核心设计是同步的、单向数据流,但现代应用中异步操作无处不在。Redux 中间件填补了这一缺口,专门解决异步流程管理、副作用隔离等复杂场景。 中间件架构原理 中间件位于 action 被发起之后、到达 reducer 之前,…...
ModBus总线协议
一、知识点 1. 什么是Modbus协议? Modbus 是一种工业通信协议,最早由 Modicon 公司在1979年提出,目的是用于 PLC(可编程逻辑控制器)之间的数据通信。它是主从式通信,即一个主机(主设备…...
【计算机网络】非阻塞IO——poll实现多路转接
🔥个人主页🔥:孤寂大仙V 🌈收录专栏🌈:计算机网络 🌹往期回顾🌹:【计算机网络】非阻塞IO——select实现多路转接 🔖流水不争,争的是滔滔不息 一、…...
在.NET Core控制器中获取AJAX传递的Body参数
.Net Core是支持前后端不分离式的开发的,如果在原始系统中采用不分离式开发,后面需要在原系统中增加功能,并且新的服务采用其他语言开发,且系统原来功能保持原样,这样前端系统可以单独调用新开发的接口。 但是&#x…...
snprintf函数用法及注意事项详解
当 format 后没有可变参数(即 ... 为空)时,va_start 的行为和后续操作如下: 1. va_start 的行为 va_start 的核心任务是根据最后一个固定参数(format)的地址,计算可变参数列表的起始位置。即使…...
vue-20(Vuex 状态管理的最佳实践)
Vuex 状态管理的最佳实践 Vuex 是管理大型 Vue.js 应用状态的一个强大工具,但其有效性取决于其组织和维护的质量。管理不善的 Vuex 存储可能会变得难以控制、难以调试,并成为性能瓶颈。本课程深入探讨构建 Vuex 存储的最佳实践,重点关注可维…...
DAX权威指南8:DAX引擎与存储优化
文章目录 十七、DAX引擎17.1 DAX 引擎的体系结构17.1.1 表格模型的双引擎架构17.1.2 存储引擎的三种模式17.1.2.1 VertiPaq引擎17.1.2.2 DirectQuery 引擎17.1.2.3 对比与最佳实践 17.1.3 数据刷新 17.2 理解 VertiPaq 存储引擎17.2.1 列式数据库17.2.2 VertiPaq 压缩17.2.2.1 …...
智慧货运飞船多维度可视化管控系统
图扑搭建智慧货运飞船可视化系统,借数字孪生技术,高精度复刻货运飞船外观、结构与运行场景。整合多维度数据,实时呈现飞行状态、设备参数等信息,助力直观洞察货运飞船运行逻辑,为航天运维、任务推演及决策提供数字化支…...
电脑开不了机,主板显示67码解决过程
文章目录 现象分析内存条问题BIOS设置问题其它问题 解决清理内存条金手指所需工具操作步骤注意事项 电脑在运行过程中,显示内存不足,重启电脑却无法启动。 现象 System Initialization 主板风扇是转的,也有灯光显示,插上屏幕&am…...
Spring Boot 类加载机制深度解析
Spring Boot 类加载机制深度解析 前言 在 Java 应用开发中,类加载机制是一个重要且复杂的话题。Spring Boot 作为现代 Java 开发的主流框架,其类加载机制更是值得深入了解。本文将从基础概念到实际应用,全面解析 Spring Boot 的类加载机制。…...
Python 训练营打卡 Day 45
TensorBoard 简单来说,TensorBoard 是 TensorFlow 自带的一个「可视化工具」,就像给机器学习模型训练过程装了一个「监控屏幕」。你可以用它直观看到训练过程中的数据变化(比如损失值、准确率)、模型结构、数据分布等,…...
自托管图书搜索引擎Bookologia
简介 什么是 Bookologia ? Bookologia 是一个专门的书籍搜索引擎,可以在几秒钟内找到任何书籍。它是开源的,可以轻松自托管在 Docker 上,为用户提供一个简单而高效的书籍查找体验。 主要特点 简洁的用户界面:界面设计…...
前端flex、grid布局
flex布局 弹性布局是指通过调整其内元素的宽高,从而在任何的显示设备上实现对可用显示空间最佳填充的能力。弹性容器扩展其内元素来填充可用空间,或将其收缩来避免溢出 简单来说,弹性盒子模型,是为了你的网页可以在不同分辨率设…...
Maven相关问题:jna版本与ES冲突 + aop失效
文章目录 1、背景2、解决3、一点思考4、环境升级导致AOP失效5、okhttp Bean找不到6、总结 记录一些Maven依赖相关的思考 1、背景 做一个监控指标收集,用一下jna依赖: <dependency><groupId>net.java.dev.jna</groupId><artifact…...
Tomcat全方位监控实施方案指南
#作者:程宏斌 文章目录 一.二进制部署1、安装包信息2、新建配置文件2.1 配置config.yaml文件2.2 上传jar包 3、修改配置3.1 备份3.2 修改bin目录下的startup.sh文件 4、重启tomcat5、访问测试 二.docker部署1、临时方案1.1、重新启动容器1.2…...
开源PHP在线客服系统源码搭建教程
在当今数字化时代,在线客服系统已成为企业与客户沟通的重要桥梁。开源PHP客服系统因其灵活性、低成本和高可定制性而受到众多企业的青睐。本文将介绍几款优秀的开源PHP客服系统,并提供详细的搭建教程。 演示网站:gofly.v1kf.com 1.1 主流开源…...
centos7升级glibic-2.28
centos7升级glibic-2.28 最近使用trae连接服务器的时候,提示远程系统不兼容: Trae CN需要glibc 2.28或更高版本。检测到的版本: 2.17。下面是升级步骤。centos7默认的glibc不支持node v18及以上。 1、进入/home/download目录(没有download,则新建一个)…...
在Docker里面运行Docker
Docker 凭借其轻量级和可移植的容器,无疑改变了软件开发和部署的世界。但如果我告诉你 Docker 本身可以在另一个 Docker 容器中运行,你会怎么想?没错!这个概念通常被称为“Docker Inside Docker”或“DinD”,它为开发人员和系统管理员开辟了一个全新的可能性领域。在这篇博…...
设计模式复习小结
1.容易忘得设计原则 接口隔离:指接口中的功能太杂则可以拆分一下。防止实现类实现了接口后自动依赖了一些不需要的功能。不同功能拆分成不同的接口。 里氏代换:强调父类能出现的地方,子类一定能正常跑。 迪米特法则:又称最少知…...
To be or Not to be, That‘s a Token——论文阅读笔记——Beyond the 80/20 Rule和R2R
本周又在同一方向上刷到两篇文章,可以说,……同学们确实卷啊,要不卷卷开放场域的推理呢? 这两篇都在讲:如何巧妙的利用带有分支能力的token来提高推理性能或效率的。 第一篇叫 Beyond the 80/20 Rule: High-Entropy Mi…...
