百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

掘金好文 | 你还在提交按钮上面用防抖函数?

itomcoil 2025-09-21 16:26 7 浏览

前言

在日常开发中会涉及到各种按钮请求,测试人员经常怼着一个按钮狂点不止。这样导致很多数据重复提交!

「前端人员经常怼着测试说」:你能不能不要点那么快啊!

(由于有时候接口返回比较慢)

「测试回怼」:我不知道点击有没有成功呀!所以多点几次咯~

「项目经理」:这是个 BUG,你们前后端开发商量一下看怎么修复吧!

「后端人员」:这不是我的问题,请求慢我没办法。。前端发生的事,找前端吧!

防抖

那既然这么喜欢围着按钮狂点,那就加防抖呗!

 //伪代码 <el-button @click='handleSubmit'>有本事再狂点击</el-button> //省略...debounce的实现 const handleSubmit = debounce(submit,1000)

加防抖看似解决了问题了。。没想到过了几天!测试又在提 BUG...

「测试」: 兄弟,我咋还是重复提交了 2 条数据啊!

「前端」:你又搞了什么骚操作?点了几下啊!

「测试」:跟上次一样啊,我作死的在点。。

「前端」:把游览器 F12 打开,我过来看一下吧!哎~

经过两人抓头排查,问题出现了:由于防抖函数设置的时间是 1 秒,后端接口返回的时间是 5 秒。。测试兄弟在这 5 秒内狂点了 100 多下。。(真是 5 秒真男人啊)

还是加按钮 Loading 吧

最后没办法了,只能通过请求之前给按钮加一个 loading 禁用状态,等接口返回 200 再清除 loading 了!

 //伪代码 <el-button @click='handleSubmit' :loading='loading'>有本事再狂点击</el-button>
let loading = false
function handleSubmit{ loading = true //开启loading ajax('xxx/xxx/xxx').then(res=>{ if(res.code == 200){ loading = false //关闭loading } }) }

这下就完美解决了测试兄弟的 BUG!可是又面临一个开发问题,就是如果很多地方要用到提交,是不是每个页面都去写一个 loading 变量啊!

接下来就是重头戏了,咱们把它封装成一个 vue 指令吧!

指令封装(vue3)

//入口文件main.jsimport { createApp } from 'vue'import App from './App.vue'

import { bLoading } from './permission/loading'
const app = createApp(App)
bLoading(app) //全局注册
//bLoading.jsimport type { App } from 'vue'
let tag =
const className = ` //loading图标的样式 el-icon { --color: inherit; -webkit-box-align: center; -ms-flex-align: center; align-items: center; display: -webkit-inline-box; display: -ms-inline-flexbox; display: inline-flex; height: 1em; -webkit-box-pack: center; -ms-flex-pack: center; justify-content: center; line-height: 1em; position: relative; width: 1em; fill: currentColor; color: var(--color); font-size: inherit; }`
//loading图标,可以去阿里图库上面复制一个svgconst i = `<i class=${className} id='loading'> <svg t="1745215287730" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2663" width="200" height="200"><path d="M834.7648 736.3584a5.632 5.632 0 1 0 11.264 0 5.632 5.632 0 0 0-11.264 0z m-124.16 128.1024a11.1616 11.1616 0 1 0 22.3744 0 11.1616 11.1616 0 0 0-22.3744 0z m-167.3216 65.8944a16.7936 16.7936 0 1 0 33.6384 0 16.7936 16.7936 0 0 0-33.6384 0zM363.1616 921.6a22.3744 22.3744 0 1 0 44.7488 0 22.3744 22.3744 0 0 0-44.7488 0z m-159.744-82.0224a28.0064 28.0064 0 1 0 55.9616 0 28.0064 28.0064 0 0 0-56.0128 0zM92.672 700.16a33.6384 33.6384 0 1 0 67.2256 0 33.6384 33.6384 0 0 0-67.2256 0zM51.2 528.9984a39.168 39.168 0 1 0 78.336 0 39.168 39.168 0 0 0-78.336 0z m34.1504-170.0864a44.8 44.8 0 1 0 89.6 0 44.8 44.8 0 0 0-89.6 0zM187.904 221.7984a50.432 50.432 0 1 0 100.864 0 50.432 50.432 0 0 0-100.864 0zM338.432 143.36a55.9616 55.9616 0 1 0 111.9232 0 55.9616 55.9616 0 0 0-111.9744 0z m169.0112-4.9152a61.5936 61.5936 0 1 0 123.2384 0 61.5936 61.5936 0 0 0-123.2384 0z m154.7776 69.632a67.1744 67.1744 0 1 0 134.3488 0 67.1744 67.1744 0 0 0-134.3488 0z m110.0288 130.816a72.8064 72.8064 0 1 0 145.5616 0 72.8064 72.8064 0 0 0-145.5616 0z m43.7248 169.472a78.3872 78.3872 0 1 0 156.8256 0 78.3872 78.3872 0 0 0-156.8256 0z" fill="" p-id="2664"></path></svg> </i>`

//核心代码export function bLoading(app: App<Element>) { app.directive('bLoading', (el, binding) => { if (typeof binding.value !== 'function') { throw new Error('Directive value must be a function') } el.addEventListener('click', => { addNode(el) //添加loading图标 setTimeout( =>{ //执行异步函数 binding.value( => { cleanNode(el) //清除loading }) }) //延迟执行异步函数 }) })}

//新增loading图标function addNode(el) { if (el.firstElementChild.tagName === 'I') { //如果按钮上面自带图标,就先移除 tag = el.firstElementChild; el.removeChild(el.firstElementChild) el.insertAdjacentHTML('afterbegin', i) } else { el.insertAdjacentHTML('afterbegin', i) } el.setAttribute('disabled', true) rotate('loading')}
//清除loading图标function cleanNode(el) { el.removeAttribute('disabled') if (el.firstElementChild) { el.removeChild(el.firstElementChild); if (tag) { el.prepend(tag) } }}
//图标旋转动画function rotate(id) { const element = document.getElementById(id); let angle = 0; // 初始角度为0度 const speed = 2; // 旋转速度,单位为度/帧(例如,每帧旋转2度) function rotate { angle = (angle + speed) % 360; // 确保角度在0到360之间循环 element.style.transform = `rotate(${angle}deg)`; // 应用旋转变换 requestAnimationFrame(rotate); // 请求下一帧的动画回调 } rotate; // 开始动画循环}

页面使用

<el-button type="primary" v-bLoading="(next) => handleSubmit(next)" :icon="Plus">有本事再狂点击</el-button>
import { Plus } from '@element-plus/icons-vue'function handleSubmit(next){ //模拟异步请求 setTimeout(=>{ //请求返回200 next //执行next函数,清除loading状态即可! },3000)}

效果

总结

通过指令的形式,解决了重复提交的问题,遏止了代码臃于,方便复用!怎么样兄弟们,如果觉得这篇文章对你有帮助,就点赞收藏吧!

点击关注公众号,“技术干货” 及时达!

相关推荐

《Queendom》宣布冠军!女团MAMAMOO四人激动落泪

网易娱乐11月1日报道据台湾媒体报道,南韩女团竞争回归的生死斗《Queendom》昨(10/31)晚播出大决赛,并以直播方式进行,6组女团、女歌手皆演唱新歌,并加总前三轮的赛前赛、音源成绩与直播现场投...

正确复制、重写别人的代码,不算抄袭

我最近在一篇文章提到,工程师应该怎样避免使用大量的库、包以及其他依赖关系。我建议的另一种方案是,如果你没有达到重用第三方代码的阈值时,那么你就可以自己编写代码。在本文中,我将讨论一个在重用和从头开始编...

HTML DOM tr 对象_html event对象

tr对象tr对象代表了HTML表格的行。HTML文档中出现一个<tr>标签,就会创建一个tr对象。tr对象集合W3C:W3C标签。集合描述W3Ccells返回...

JS 打造动态表格_js如何动态改变表格内容

后台列表页最常见的需求:点击表头排序+一键全选。本文用原生js代码实现零依赖方案,涵盖DOM查询、排序算法、事件代理三大核心技能。效果速览一、核心思路事件入口:为每个<th>绑...

连肝7个晚上,总结了66条计算机网络的知识点

作者|哪吒来源|程序员小灰(ID:chengxuyuanxiaohui)计算机网络知识是面试常考的内容,在实际工作中经常涉及。最近,我总结了66条计算机网络相关的知识点。1、比较http0....

Vue 中 强制组件重新渲染的正确方法

作者:MichaelThiessen译者:前端小智来源:hackernoon有时候,依赖Vue响应方式来更新数据是不够的,相反,我们需要手动重新渲染组件来更新数据。或者,我们可能只想抛开当前的...

为什么100个前端只有1人能说清?浏览器重排/重绘深度解析

面试现场的"致命拷问""你的项目里做过哪些性能优化?能具体讲讲重排和重绘的区别吗?"作为面试官,我在秋招季连续面试过100多位前端候选人,这句提问几乎成了必考题。但令...

HTML DOM 介绍_dom4j html

HTMLDOM(文档对象模型)是一种基于文档的编程接口,它是HTML和XML文档的编程接口。它可以让开发人员通过JavaScript或其他脚本语言来访问和操作HTML和XML文档...

JavaScript 事件——“事件流和事件处理程序”的注意要点

事件流事件流描述的是从页面中接收事件的顺序。IE的事件流是事件冒泡流,而NetscapeCommunicator的事件流是事件捕获流。事件冒泡即事件开始时由最具体的元素接收,然后逐级向上传播到较为不...

探秘 Web 水印技术_水印制作网页

作者:fransli,腾讯PCG前端开发工程师Web水印技术在信息安全和版权保护等领域有着广泛的应用,对防止信息泄露或知识产品被侵犯有重要意义。水印根据可见性可分为可见水印和不可见水印(盲水印)...

国外顶流网红为流量拍摄性侵女学生?仅被封杀三月,回归仍爆火

曾经的油管之王,顶流网红DavidDobrik复出了。一切似乎都跟他因和成员灌酒性侵女学生被骂到退网之前一样:住在950万美元的豪宅,开着20万美元的阿斯顿马丁,每条视频都有数百万观看...人们仿佛...

JavaScript 内存泄漏排查方法_js内存泄漏及解决方法

一、概述本文主要介绍了如何通过Devtools的Memory内存工具排查JavaScript内存泄漏问题。先介绍了一些相关概念,说明了Memory内存工具的使用方式,然后介绍了堆快照的...

外贸独立站,网站优化的具体内容_外贸独立站,网站优化的具体内容有哪些

Wordpress网站优化,是通过优化代码、数据库、缓存、CSS/JS等内容,提升网站加载速度、交互性和稳定性。网站加载速度,是Google搜索引擎的第一权重,也是SEO优化的前提。1.优化渲染阻塞。...

这8个CSS工具可以提升编程速度_css用什么编译器

下面为大家推荐的这8个CSS工具,有提供函数的,有提供类的,有提取代码的,还有收集CSS的统计数据的……请花费两分钟的时间看完这篇文章,或许你会找到意外的惊喜,并且为你的编程之路打开了一扇新的大门。1...

vue的理解-vue源码 历史 简介 核心特性 和jquery区别 和 react对比

一、从历史说起Web是WorldWideWeb的简称,中文译为万维网我们可以将它规划成如下的几个时代来进行理解石器时代文明时代工业革命时代百花齐放时代石器时代石器时代指的就是我们的静态网页,可以欣...