JS前端无侵入实现防止重复提交请求技术
JS前端无侵入实现防止重复提交请求技术
最近在代码发布测试的过程中,我发现有些请求非常的消耗服务器资源,而系统测试人员因为响应太慢而不停的点击请求。我是很看不惯系统存在不顺眼的问题,做事喜欢精益求精,也很喜欢和别人争论技术,有时候硬要争得你死我活。
实在看不下去系统存在这个问题,下定决心好好整改一波,我们系统几乎都是使用Ajax技术发请求,使用的是针对我们系统特点对jQuery的post方法进行了封装的newpost方法,这个方法扩展了异常响应,登陆超时,token验证,加载时渲染等自动完成功能,参数不变,也是我写的。由于系统模块很多,功能很强大,代码也很多,为了统一使用此方法,花了不少时间将post修改成newpost。这次为了限制请求次数这个小功能,我请示了领导并把初步的方案说与他听,他同意了我做此次修改。
我调查了几个有防止请求重发控制的页面,几乎都是发起请求后到响应这段时间对按钮置灰警用处理,这种方案有很多的缺点,如:
1. 长时间未响应的请求给人不友好的体验
2. 需要针对每个按钮做请求前后的控制,会因为处理各种小问题而变得难以维护
3. 业务与控制逻辑混合在一起使得函数功能不单一
我最初的想法就是想找到一种类似于面向切面的无侵入页面的方式实现请求限制,因此想到了如下方案:
1. 只控制用户最后一次的请求
2. 第一次请求未响应之前一分钟之内不发相同的请求
3. 如果前一次请求得到了响应,那么第二次相同的请求也不做限制
后来发现这种方案行不通,主要有以下几个方面的原因:
1. 大多数页面每交互一次都会发起多个不同的请求。
2. 没个请求响应的时间也不一样,可能后发的请求先响应
我把这种想法告诉了导师,经过一番争论后认为我无法通过不侵入的方式实现这样的功能,按照他的想法,我得修改整个项目有请求的代码,完全低估了我对JS的了解,后来我独自优化了方案,不在只控制最后一个请求,代码实现的大纲如下:
发起POST请求前处理逻辑
1. 获取全局上下文中的存储请求信息的数组变量
2. 遍历请求响应数组
3. 清理大于两分钟未响应的请求,防止页面长时间不响应
4. 找到了与本次相同的请求
a) 如果请求小于200ms,则判断为系统行为,直接将环境和回调函数压入已有请求的调用栈,否则替换之前的请求的调用栈
b) 不发请求
5. 没有找到与本次相同的请求
a) 将请求信息、时间、调用环境和回调函数压进请求响应数组,请求URL和条件con作为唯一标识,调用栈也是一个由当前环境和回调函数组成的数组,使用数组解决相同的请求(url和con都一样)
b) 发起请求
请求响应后处理逻辑
1. 获取全局上下文中的存储请求信息的数组变量
2. 遍历请求响应数组
3. 找到了与此响应相同的请求
a) 遍历此请求的调用栈并依次调用回调函数
b) 删除已经处理的请求
4. 没有找到与此响应相同的请求则什么都不做
具体实现:
/*@description 自定义事件
*@method_costomEvent
*@formuphy
*/
muphypost: function (url, condition, callback, type, option) {
if (typeof condition === 'function') {
option = type;
type = callback;
callback = condition;
condition = {};//jQuery:condition = undefine;
}
let that = this, reqUrl = (url + JSON.stringify(condition)).substr(0, 1000);
//判断是否需要发起请求
let hasReq = beforePost();
if (hasReq === true) {
return that;
}
//新增请求Token
muphy.addToken(condition);
//借助jQuery发请求
$.post(url, condition, function (data, status){
afterPost(data, status);
}, type);
return that;
//请求前处理函数
function beforePost() {
//获取历史请求
let postHistory = muphy.postHistory, now = new Date();
//遍历请求响应数组,找到与此请求相同的请求
for (let i = 0; i < postHistory.length;i++) {
if (now - postHistory[i]._startTime > 120000) {
//清理大于两分钟未响应的请求
postHistory.splice(i, 1);
} elseif (postHistory[i]._reqUrl === reqUrl){
//找到了就更新请求时间,并将执行环境和回调函数放入请求响应数组的调用栈;
//小于200ms判定为系统行为直接将环境和回调函数压入已有请求的调用栈,
//否则替换之前的请求
if (now - postHistory[i]._startTime < 200) {
postHistory[i]._startTime =now;
postHistory[i]._handles.push({ _callback: callback, _this: that });
} else {
postHistory[i]._handles =[{ _callback: callback, _this: that }];
}
//返回true 表示本次不发请求
return true;
}
};
//没找到与此相同的请求时将请求信息、时间、调用环境和回调函数压进请求响应数组
postHistory.push({ _reqUrl: reqUrl,_startTime: new Date(), _handles: [{ _callback:callback, _this: that }] });
//返回false 表示本次需要发请求
return false;
}
//请求后处理函数
function afterPost(data, status) {
//获取请求响应数组
let postHistory = muphy.postHistory;
//遍历请求响应数组,找到与此请求相同的请求
for (let i = 0; i < postHistory.length;i++) {
if (postHistory[i]._reqUrl === reqUrl) {
//找到相同的请求,遍历此请求的调用栈并调用
muphy.each(postHistory[i]._handles, function () {
this._callback.call(this._this, data, status);
});
//调用完成删除已经处理的请求
postHistory.splice(i, 1);
break;
}
}
}
}
JS前端无侵入实现防止重复提交请求技术的更多相关文章
- ajax防止重复提交请求1
ajax防止重复提交请求 A. 独占型提交 只允许同时存在一次提交操作,并且直到本次提交完成才能进行下一次提交. module.submit = function() { if (this.pro ...
- JS中如何防止表单重复提交问题
在登录页面html中写如下代码 <script type="text/javascript"> var issubmit=false; function dosubmi ...
- 使用js是想防止表单重复提交的效果
直接上代码: <html> <head> <title>Form表单</title> <script type="text/javasc ...
- js 前端向服务器端传送文件的常用请求方式
在做项目的过程当中写到文件上传的功能,想着之前也是踩坑过来的,就在这里总结下自己常用的方法吧.我们现在一般都是通过ajax来搭起前后端数据交互的桥梁,但是大家在做到有文件需要上传的时候就会发现我们用a ...
- java web解决表单重复提交问题
我们大家再进行web开发的时候,必不可少会遇见表单重复提交问题.今天就来给总结如何解决表单提交问题,欢迎大家交流指正. 首先我们在讨论如何解决表单重复提交问题之前先来解决三个问题:1.什么叫表单重复提 ...
- java web解决表单重复提交
首先我们在讨论如何解决表单重复提交问题之前先来解决三个问题:1.什么叫表单重复提交?2.什么情况下会出现表单重复提交?3.什么情况需要避免表单重复提交? 什么叫表单提交问题,说白了,就是同一份信息,重 ...
- php解决表单重复提交
php解决表单重复提交时间:2015-2-28 | 评论:1条评论 | 被查看了 189 次 | 标签:php, W3cui重复提交是我们开发中会常碰到的一个问题,除了我们使用js来防止表单的重复提交 ...
- PHP防止表单重复提交的解决方法
PHP+SESSION防止表单重复提交 index.php 当前表单页面is_submit设为0 SESSION_START(); $_SESSION['is_submit'] = 0; <fo ...
- php 解决表单重复提交实现方法介绍
重复提交是我们开发中会常碰到的一个问题,除了我们使用js来防止表单的重复提交,同时还可以使用php来防止重复提交哦. 例1 代码如下 复制代码 <?php/* * php中如何防止表单的重复提 ...
随机推荐
- SLAM+语音机器人DIY系列:(二)ROS入门——1.ROS是什么
摘要 ROS机器人操作系统在机器人应用领域很流行,依托代码开源和模块间协作等特性,给机器人开发者带来了很大的方便.我们的机器人“miiboo”中的大部分程序也采用ROS进行开发,所以本文就重点对ROS ...
- 使用.NetCore 控制台演示 熔断 降级(polly)
1.熔断降级的概念: 熔断:我这里有一根长度一米的钢铁,钢铁的熔点1000度(假设),现在我想用力把这根钢铁折弯,但是人的力有限达不到折弯的点,然后我使用火给钢铁加热,每隔一段时间我就会尝试一下是否能 ...
- ASP.NET应用程序服务器集群方案
本文采用Nginx来实现ASP.NET程序集群化. 准备环境 首先准备Nginx环境,Windows版本下载链接:http://nginx.org/en/download.html 解压后文件格式如下 ...
- mac 卸载通过官网下载包安装的node
sudo rm -rf /usr/local/{bin/{node,npm},lib/node_modules/npm,lib/node,share/man/*/node.*}
- 第十一课 CSS介绍与font字体 css学习1
一.CSS样式规则 1.基本结构 <html> <head> <style> h1{ color: orange; } </style> </he ...
- 在Windows Phone 8.1中使用Sqlite数据库
前言 我的工作目前不涉及到Windows phone的开发,但是业余时间也开发过几款app.以前由于各种条件的限制,只接触到WP8.0设备的app开发. 最近几个月开始将WP8的应用迁移到WP8.1, ...
- 简单易懂的单元测试框架-gtest(一)
简介 gtest是google开源的一个单元测试框架,以其简单易学的特点被广泛使用.该框架以第三方库的方式插入被测代码中.同其他单元测试框架相似,gtest也通过制作测试样例来进行代码测试.同 ...
- django mongodb配置
#settings.py import os from mongoengine import * BASE_DIR = os.path.dirname(os.path.dirname(os.path. ...
- sql语句修改字段类型和增加字段
/*修改字段类型*/ ) go /*增加字段和说明*/ ) EXECUTE sp_addextendedproperty N'MS_Description','说明文字',N'user',N'dbo' ...
- Webdriver之API详解(2)
前言:今天继续上一篇文章https://www.cnblogs.com/linuxchao/p/linuxchao-selenium-apione.html分享selenium' webdriver ...