前端在向后端进行数据提交的时候,一般会须要在第一次提交返回前,阻止用户在快速点击发送二次请求,即防止重复提交,最简单的方法是使用标志参数或者 class 元素控制,但缺点是,每一个控制重复提交的地方都须要加上这个逻辑,重复性太强,且控制逻辑不统一。html
目前前端使用的是http协议,因此提交方式为两种前端
经过 jQuery 提供的 ajaxPrefilter 方法,将在请求提交以前进行过滤,仅保留第一次请求,后续的请求 abort 阻止掉,具体实现代码以下jquery
/**
* _pendingRequests = {
* 'http:xxx.xxxx.do':['domain=P2P','xxxx=aaa'],
* 'http:xxx.yyyy.do':['domain=P3P','xxxx=bbb']
* }
* 该对象的 key 是请求的 url ,value 是由请求参数转化成的字符串数组
*/
var _pendingRequests = {};
$.ajaxPrefilter(function(options, originalOptions, jqXHR) {
var p_item = { //保存请求请求的url
key:options.url,
index:0
},
dataArray = options.data ? options.data.split('&') : [];
compareData = function(beforD,afterD) {
//当url相同时,以此比较保存的参数对象,若参数对象相同,则返回false,若第一个就相同,则跳出循环
// 反之说明当前参数对象列表中没有与将要提交的参数相同,则可看为不一样的请求,返回true,容许发起请求
var result = false;
for(var i=0;i<beforD.length;i++){
if(beforD[i]){
result = false;
var beforData = beforD[i];
for(var j=0;j<beforData.length;j++){
if(afterD[j] !== beforData[j]){
result = true;
break;
}
}
if (!result){
break;
}
}else {
result = true;
continue;
}
}
return result;
};
//若请求队列中不存在或者同一个请求不一样参数,且不为html后缀,则加入队列中
if (( !_pendingRequests[p_item.key] || compareData(_pendingRequests[p_item.key],dataArray) ) && p_item.key.indexOf('.html') === -1) {
//给 index 赋值是由于请求是异步返回的,index用于标记第一个请求
if(_pendingRequests[p_item.key]){
p_item.index = _pendingRequests[p_item.key].push(dataArray)-1;
} else{
_pendingRequests[p_item.key] = [dataArray];
p_item.index = 0;
}
} else if (p_item.key.indexOf('.html') === -1) {
jqXHR.abort(); // 放弃后触发的重复提交,仅保留第一次提交
//pendingRequests[key].abort(); // 放弃先触发的提交
}
var complete = options.complete;
//请求完成
options.complete = function(jqXHR, textStatus) {
// 经过 key 和 index 获取成功返回的请求,将其值为 null ,下一次该请求即是在请求队列中即是新的一个请求
_pendingRequests[p_item.key][p_item.index] = null;
if ($.isFunction(complete)) {
complete.apply(this, arguments);
}
};
});
复制代码
表单的处理就稍微要麻烦点,但大体思路和异步的相同,等待第一次请求返回的同时,阻止后续触发的请求发送 首先基于jquery扩展了一个自定义的方法,以下后端
$.fn.preventDoubleSubmission = function() {
$(this).on('submit', function(e) {
var $form = $(this);
// $form.data('submitted') 经过该变量判断请求的状态
if ($form.data('submitted') === true) {
//阻止请求
e.preventDefault();
} else {
$form.data('submitted', true);
if ($form.attr('target') === '_blank') {
setTimeout(function() {
$form.data('submitted', false);
}, 800);
}
}
});
return this;
};
复制代码
当表单初次提交时,经过 jQuery.data() 设置一个标志位,当表单重复提交时,判断设置的标志位,如果提交状态,将阻止表单的提交事件。当form.target = _blank 提交后打开新界面的状况,将在800毫秒后恢复原界面表单可提交状态。api
为了方便对全站的表单提交统一处理,可对须要放重复提交的表单添加一个class preventDouble
,在页面渲染后,统一加上事件监听数组
//扫描带有 preventDouble 标识的form表单
$(function() {
var f = $('.contain form.preventDouble');
for (var i=0;i<f.length;i++){
$(f[i]).preventDoubleSubmission();
}
});
复制代码
小贴士bash
提交按钮需使用
type=’submit’
,由于监听的是表单的submit事件 不建议屡次监听submit事件,会致使放重复提交失效 在表单提交前一般会有些表单检验的操做,因此当校验失败的时候,能够经过event.preventDefault()
阻止表单提交app