防止 javascript 重复点击和提交的关键方法有三种:1. 禁用按钮法,点击后立即禁用按钮并更改文本提示,请求完成后恢复;2. 节流函数(throttle),限制函数在设定时间间隔内仅执行一次,适用于非即时响应场景;3. 使用标志位,通过变量标记操作状态以阻止重复触发。此外,前端可通过按钮文本变化或加载动画提升用户提示体验,而后端则可通过 token 机制、悲观锁/乐观锁及幂等性设计确保数据一致性与安全性。选择方案应根据具体场景决定,前端与后端结合使用可提供更全面的防护。

防止 JavaScript 重复点击和提交,关键在于控制用户行为,避免短时间内多次触发相同操作。这不仅仅是用户体验的问题,更关系到数据一致性和服务器压力。

解决方案

  1. 禁用按钮法: 这是最直接也最常用的方法。点击按钮后,立即禁用它,防止用户再次点击。待请求完成后,再重新启用按钮。

const button = document.getElementById('myButton');button.addEventListener('click', function() {button.disabled = true;button.textContent = '处理中...'; // 可选:更改按钮文本提示用户// 模拟一个异步请求setTimeout(() => {// 请求成功或失败后的处理button.disabled = false;button.textContent = '提交'; // 恢复按钮文本}, 2000);
});

 

 这种方法的优点是简单易懂,缺点是用户体验可能略差,因为按钮禁用期间无法进行任何操作。可以结合按钮文本的变化来改善用户体验。

2,节流函数(Throttle): 使用节流函数可以限制函数在一定时间内只能执行一次。这适用于那些不需要立即响应的场景,比如搜索框的输入事件。 

function throttle(func, delay) {let timeoutId;let lastExecTime = 0;return function(...args) {const now = Date.now();const timeSinceLastExec = now - lastExecTime;if (!timeoutId) {if (timeSinceLastExec >= delay) {func.apply(this, args);lastExecTime = now;} else {timeoutId = setTimeout(() => {func.apply(this, args);lastExecTime = Date.now();timeoutId = null;}, delay - timeSinceLastExec);}}};
}const myFunc = () => {console.log("函数执行");
};const throttledFunc = throttle(myFunc, 1000); // 1秒内只允许执行一次// 多次调用 throttledFunc,但实际上只有在间隔超过 1 秒时才会执行
throttledFunc();
throttledFunc();
setTimeout(throttledFunc, 500);
setTimeout(throttledFunc, 1200);

 

节流函数的核心在于时间控制,确保函数在规定的时间间隔内只执行一次。

3,使用标志位: 定义一个变量来标记是否正在进行操作。如果正在进行操作,则忽略后续的点击事件

let isSubmitting = false;const form = document.getElementById('myForm');form.addEventListener('submit', function(event) {event.preventDefault(); // 阻止默认的表单提交if (isSubmitting) {console.log('正在提交,请稍候...');return;}isSubmitting = true;console.log('开始提交...');// 模拟一个异步提交setTimeout(() => {console.log('提交完成!');isSubmitting = false;}, 3000);
});

 

这种方法简单直接,但需要注意在异步操作完成后,一定要将标志位重置,否则后续操作将无法进行。

如何选择最合适的防止重复提交方案?

这取决于你的具体应用场景。如果只需要简单地防止按钮被连续点击,禁用按钮法是最简单的选择。如果需要限制函数在一定时间内只能执行一次,节流函数更合适。如果需要更精细的控制,比如在表单提交过程中防止重复提交,使用标志位可能更灵活。

禁用按钮后,如何优雅地提示用户正在处理中?

仅仅禁用按钮可能让用户感到困惑,不知道发生了什么。更好的做法是结合按钮文本的变化,例如将按钮文本改为“处理中...”或显示一个加载动画。此外,还可以使用模态框或消息提示来告知用户当前的状态。

const button = document.getElementById('myButton');
const originalText = button.textContent;button.addEventListener('click', function() {button.disabled = true;button.textContent = '处理中...'; // 更改按钮文本button.classList.add('loading'); // 添加一个 loading class,用于显示加载动画// 模拟一个异步请求setTimeout(() => {// 请求成功或失败后的处理button.disabled = false;button.textContent = originalText; // 恢复按钮文本button.classList.remove('loading'); // 移除 loading class}, 2000);
});

 在 CSS 中定义 .loading 类,例如:

.loading {position: relative;padding-right: 25px; /* 为加载动画预留空间 */
}.loading::after {content: '';position: absolute;top: 50%;right: 5px;width: 15px;height: 15px;border-radius: 50%;border: 2px solid #ccc;border-top-color: #333;animation: spin 1s linear infinite;transform: translateY(-50%);
}@keyframes spin {to {transform: translateY(-50%) rotate(360deg);}
}

 

除了前端,后端如何防止重复提交?

前端的防止重复提交只能解决一部分问题,更可靠的方案是在后端进行验证。常见的后端防止重复提交的方法包括:

  • Token 机制: 在前端页面生成一个唯一的 Token,每次提交请求时都带上这个 Token。后端验证 Token 的有效性,如果 Token 已经被使用过,则拒绝本次请求。
  • 悲观锁/乐观锁: 在数据库层面使用锁机制,确保同一时间只有一个请求可以修改数据。
  • 幂等性设计: 将接口设计成幂等的,即使多次调用,结果也应该是一致的。例如,可以使用唯一 ID 来标识请求,后端根据 ID 来判断是否已经处理过该请求。

后端防止重复提交是更可靠的方案,可以有效地防止恶意攻击和网络延迟等问题导致的重复提交。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/86449.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/86449.shtml
英文地址,请注明出处:http://en.pswp.cn/pingmian/86449.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【信创-k8s】银河麒麟V10国防版+鲲鹏/飞腾(arm64架构)在线/离线部署k8s1.30+kubesphere

银河麒麟作为国家核高基专项的重要成果,国防版凭借其卓越的安全性和可靠性,已成为军工领域的首选操作系统。之前我们在适配麒麟V4国防版的过程中已发现诸多安全性要求,而麒麟V10国防版在安全防护等级上又达到了更高的级别。 本文将主要演示离…

解锁单周期MIPS硬布线:Logisim实战全攻略

目录 一、引言二、MIPS 架构与单周期设计原理2.1 MIPS 架构概述2.2 单周期设计原理剖析 三、Logisim 工具基础3.1 Logisim 简介3.2 基本操作与组件认识 四、单周期 MIPS 硬布线设计步骤4.1 了解 MIPS 指令集4.2 搭建数据通路4.3 设计硬布线控制器4.4 在 Logisim 中创建电路 五、…

7.4.2B+树

B树: (1)每个分支节点最多有m个子树(孩子节点)。 阶:即看当前的B树是几阶B树,就看每个分支节点最多有几个子树,还是看最下一层有几个分叉就是几阶??? 叶子节点:最下边的一层叫叶子…

MFC获取本机所有IP、局域网所有IP、本机和局域网可连接IP

获取本机所有IP地址 // 获取本机所有IP地址 int CMachine::GetLocalIPs(std::vector<CString>& vIPValue) {//返回IP数量&#xff0c; -1表示获取失败vIPValue.clear();int IpNum 0;//1.初始化wsa WSADATA wsaData;int ret WSAStartup(MAKEWORD(2, 2), &wsaD…

【C语言】贪吃蛇小游戏

1. 所需知识 C语言函数、枚举、结构体、动态内存管理、预处理指令、链表、Win32 API... 2. Win32 API介绍 2.1 Win32 API windows这个多作业系统除了协调应用程序的执行、分配内存、管理资源之外&#xff0c;它同时也是一个很大的服务中心&#xff0c;调用这个服务中心的各种…

PostgreSQL 容器化分布式技术方案

&#x1f4cb; 目录 引言&#xff1a;为什么选择容器化PostgreSQLPostgreSQL容器化基础分布式架构设计高可用实现方案读写分离架构动态扩缩容策略生产环境实践总结与展望 引言&#xff1a;为什么选择容器化PostgreSQL 在数字化转型的浪潮中&#xff0c;数据库作为企业的"…

NV025NV033美光固态闪存NV038NV040

美光固态闪存技术突破与市场布局深度解析 一、技术突破&#xff1a;232层NAND闪存与高密度存储的革新 美光NV系列固态闪存的核心竞争力源于其232层NAND闪存技术&#xff0c;这一技术通过垂直堆叠工艺&#xff0c;将存储单元层层叠加&#xff0c;宛如在指甲盖面积内构建超过20…

Matplotlib 绘图库从入门到精通:Python 数据可视化全解析

引言 在数据科学的世界里&#xff0c;"一图胜千言" 这句话有着深刻的含义。数据可视化不仅是数据分析师展示成果的重要手段&#xff0c;更是数据科学家探索数据、发现规律的强大工具。Matplotlib 作为 Python 生态系统中最著名的数据可视化库&#xff0c;为我们提供…

北斗导航 | 基于CNN-LSTM-PSO算法的接收机自主完好性监测算法

接收机自主完好性监测 原理概述1. 算法架构2. 核心创新点3. 工作流程数学模型1. CNN特征提取2. LSTM时序建模3. PSO优化决策MATLAB完整代码算法优势性能对比应用场景扩展方向原理概述 1. 算法架构 #mermaid-svg-fITV6QrXL1fNYFwG {font-family:"trebuchet ms",verda…

【微信小程序】9、用户拒绝授权地理位置后再次请求授权

1、获取用户当前的地理位置 在本专栏的上一篇文章中讲了如何 获取用户当前的地理位置 首次请求 wx.getLocation API 后&#xff0c;会拉起用户授权界面 但这时用户可能会拒绝授权&#xff0c;当你再次请求 wx.getLocation API 后&#xff0c;没有任何效果。 2、打开设置 用…

嵌入式Linux驱动开发基础-1 hello驱动

1:APP打开的文件在内核中如何表示 1.1 APP 打开文件时&#xff0c;可以得到一个整数&#xff0c;这个整数被称为文件句柄。对于 APP 的每一个文件句柄&#xff0c;在内核里面都有一个“struct file ”与之对应 当我们使用 open 打开文件时&#xff0c;传入的 flags 、 mode…

目标跟踪存在问题以及解决方案

3D 跟踪 一、数据特性引发的跟踪挑战 1. 点云稀疏性与远距离特征缺失 问题表现&#xff1a; 激光雷达点云密度随距离平方衰减&#xff08;如 100 米外车辆点云数不足近距离的 1/10&#xff09;&#xff0c;导致远距离目标几何特征&#xff08;如车轮、车顶轮廓&#xff09;不…

JavaSE-JDK安装

目录 一.在官网下载安装包 二.安装JDK 三.检测JDK是否安装成功 四.配置系统环境变量 一.在官网下载安装包 Oracle官网https://www.oracle.com/cn/java/technologies/downloads/ 二.安装JDK 1.首先在C盘以为的其他盘中创建一个自己可以找到的存放JDK路径&#xff1a; 2.双击下…

使用docker搭建redis主从架构,一主2从

使用Docker搭建Redis主从架构&#xff08;一主两从&#xff09; Redis主从架构是提高系统可用性和读取性能的重要方案&#xff0c;通过Docker可以快速搭建该架构。下面将详细介绍搭建步骤。 架构设计 我们将搭建包含以下组件的架构&#xff1a; 1个主节点&#xff08;Maste…

机器学习3——参数估计之极大似然估计

参数估计 问题背景&#xff1a; P ( ω i ∣ x ) p ( x ∣ ω i ) P ( ω i ) p ( x ) p ( x ) ∑ j 1 c p ( x ∣ ω j ) P ( ω j ) \begin{aligned} & P\left(\omega_i \mid \mathbf{x}\right)\frac{p\left(\mathbf{x} \mid \omega_i\right) P\left(\omega_i\right)…

Spring AOP Pointcut 表达式的语法是怎样的?(execution(...) 是最常用的,还有哪些

Pointcut 表达式是 AOP 的核心&#xff0c;我将详细解析最常用的 execution 表达式&#xff0c;并介绍其他几种同样非常有用的表达式。 1. execution 指示符 (最常用&#xff0c;最强大) execution 用于匹配方法的执行&#xff08;Join Point&#xff09;。它的语法结构最为完…

基于 SpringBoot+Vue 的台球厅管理系统的设计与实现(毕业论文)

基于 SpringBootVue 的台球厅管理系统的设计与实现&#xff08;模板&#xff09;[三号宋体加粗&#xff0c;居中] 摘 要[首行缩进2字符&#xff0c;五号黑体加粗]&#xff1a;摘要内容[五号楷体]本文所提出的基于J2EE/EJB标准的电子化采购平台及其CRM组件综合解决方案&#xf…

运营医疗信息化建设的思路

医疗机构加强运营管理&#xff0c;必须依赖强有力的医院信息系统。信息化很重要&#xff0c;但不能为了信息化而信息化。运营信息化必须有明确的建设目标。 运营信息化建设的目标&#xff0c;包括几个方面&#xff1a; 1.实时反映业务&#xff1b; 2.体现内控思维&#xff1b…

6.24_JAVA_微服务day07_RabbitMQ高级

1、 RabbitListener(queuesToDeclare/*此处是固定写法&#xff0c;只能写这个玩意儿&#xff0c;因为这里是库里的方法*/ Queue(name "lazy.queue",//如果不存在就创建lazy.queue队列durable "true",//把耐用打开arguments Argument(name "x-que…

Python打卡:Day38

知识点回顾&#xff1a; Dataset类的__getitem__和__len__方法&#xff08;本质是python的特殊方法&#xff09;Dataloader类minist手写数据集的了解 浙大疏锦行