当前位置:Gxlcms > JavaScript > 前端防止用户重复提交js实现代码示例

前端防止用户重复提交js实现代码示例

时间:2021-07-01 10:21:17 帮助过:1人阅读

背景

前端在向后端进行数据提交的时候,通常会需要在第一次提交返回前,阻止用户在快速点击发送二次请求,即防止重复提交,最简单的方法是使用标志参数或者 class 元素控制,但缺点是,每个控制重复提交的地方都需要加上这个逻辑,重复性太强,且控制逻辑不统一。

目前前端使用的是http协议,所以提交方式为两种

  • 异步提交,使用jQuery.ajax()
  • form 表单同步提交

下面这篇文章将给大家详细介绍关于这两种方法实现的方法示例,下面话不多说了,来随着小编一起看看详细的介绍吧

异步防重复提交的方案如下

通过 jQuery 提供的 ajaxPrefilter 方法,将在请求提交之前进行过滤,仅保留第一次请求,后续的请求 abort 阻止掉,具体实现代码如下

  1. /**
  2. * _pendingRequests = {
  3. * 'http:xxx.xxxx.do':['domain=P2P','xxxx=aaa'],
  4. * 'http:xxx.yyyy.do':['domain=P3P','xxxx=bbb']
  5. * }
  6. * 该对象的 key 是请求的 url ,value 是由请求参数转化成的字符串数组
  7. */
  8. var _pendingRequests = {};
  9. $.ajaxPrefilter(function(options, originalOptions, jqXHR) {
  10. var p_item = { //保存请求请求的url
  11. key:options.url,
  12. index:0
  13. },
  14. dataArray = options.data ? options.data.split('&') : [];
  15. compareData = function(beforD,afterD) {
  16. //当url相同时,以此比较保存的参数对象,若参数对象相同,则返回false,若第一个就相同,则跳出循环
  17. // 反之说明当前参数对象列表中没有与将要提交的参数相同,则可看为不同的请求,返回true,允许发起请求
  18. var result = false;
  19. for(var i=0;i<beforD.length;i++){
  20. if(beforD[i]){
  21. result = false;
  22. var beforData = beforD[i];
  23. for(var j=0;j<beforData.length;j++){
  24. if(afterD[j] !== beforData[j]){
  25. result = true;
  26. break;
  27. }
  28. }
  29. if (!result){
  30. break;
  31. }
  32. }else {
  33. result = true;
  34. continue;
  35. }
  36. }
  37. return result;
  38. };
  39. //若请求队列中不存在或者同一个请求不同参数,且不为html后缀,则加入队列中
  40. if (( !_pendingRequests[p_item.key] || compareData(_pendingRequests[p_item.key],dataArray) ) && p_item.key.indexOf('.html') === -1) {
  41. //给 index 赋值是因为请求是异步返回的,index用于标记第一个请求
  42. if(_pendingRequests[p_item.key]){
  43. p_item.index = _pendingRequests[p_item.key].push(dataArray)-1;
  44. } else{
  45. _pendingRequests[p_item.key] = [dataArray];
  46. p_item.index = 0;
  47. }
  48. } else if (p_item.key.indexOf('.html') === -1) {
  49. jqXHR.abort(); // 放弃后触发的重复提交,仅保留第一次提交
  50. //pendingRequests[key].abort(); // 放弃先触发的提交
  51. }
  52. var complete = options.complete;
  53. //请求完成
  54. options.complete = function(jqXHR, textStatus) {
  55. // 通过 key 和 index 获取成功返回的请求,将其值为 null ,下一次该请求便是在请求队列中便是新的一个请求
  56. _pendingRequests[p_item.key][p_item.index] = null;
  57. if ($.isFunction(complete)) {
  58. complete.apply(this, arguments);
  59. }
  60. };
  61. });

jquery.ajaxprefilter官方文档

表单提交防重复提交的方案如下

表单的处理就稍微要麻烦点,但大致思路和异步的相同,等待第一次请求返回的同时,阻止后续触发的请求发送
首先基于jquery扩展了一个自定义的方法,如下

  1. $.fn.preventDoubleSubmission = function() {
  2. $(this).on('submit', function(e) {
  3. var $form = $(this);
  4. // $form.data('submitted') 通过该变量判断请求的状态
  5. if ($form.data('submitted') === true) {
  6. //阻止请求
  7. e.preventDefault();
  8. } else {
  9. $form.data('submitted', true);
  10. if ($form.attr('target') === '_blank') {
  11. setTimeout(function() {
  12. $form.data('submitted', false);
  13. }, 800);
  14. }
  15. }
  16. });
  17. return this;
  18. };

当表单初次提交时,通过 jQuery.data() 设置一个标志位,当表单重复提交时,判断设置的标志位,若是提交状态,将阻止表单的提交事件。当form.target = _blank 提交后打开新界面的情况,将在800毫秒后恢复原界面表单可提交状态。

为了方便对全站的表单提交统一处理,可对需要放重复提交的表单添加一个class preventDouble,在页面渲染后,统一加上事件监听

  1. //扫描带有 preventDouble 标识的form表单
  2. $(function() {
  3. var f = $('.contain form.preventDouble');
  4. for (var i=0;i<f.length;i++){
  5. $(f[i]).preventDoubleSubmission();
  6. }
  7. });

小贴士

  • 提交按钮需使用 type='submit' ,因为监听的是表单的submit事件
  • 不建议多次监听submit事件,会导致放重复提交失效
  • 在表单提交前通常会有些表单检验的操作,所以当校验失败的时候,可以通过 event.preventDefault() 阻止表单提交

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

人气教程排行