背景

表单重复提交会造成数据重复,增加服务器负载,严重甚至会造成服务器宕机等情况,有效防止表单重复提交有一定的必要性。
常见的防止表单重复提交解决方案有以下几种:

一、通过一个标识来控制表单提交之后,再次提交会直接返回处理

示例:

<html>
<head><title>防止表单重复提交</title>
</head>
<body><form action="/path/post" onsubmit="return dosubmit()" method="post"><input type="submit" value="提交" id="submit"></form><script type="text/javascript">//默认提交状态为falselet isCommitted = false;function dosubmit(){if(isCommitted == false){//提交表单后,讲提交状态改为truecommitStatus = true;//返回true,让表单正常提交return true;}else{return false;}}
</script>
</body>
</html>

注意:通过js代码,当用户点击提交按钮后,屏蔽提交按钮使用户无法点击提交按钮或点击无效,从而实现防止表单重复提交。

二、通过点击提交一次按钮之后,将该按钮设置为不可用处理

示例:

<html>
<head><title>防止表单重复提交</title>
</head>
<body><form action="/path/post" onsubmit="return dosubmit()" method="post"><input type="submit" value="提交" id="submit"></form><script type="text/javascript">function dosubmit() {//获取表单提交按钮Var btnSubmit = documen.getElementById("sumit");//将表单提交按钮设置为不可用,可以避免用户再次点击提交按钮进行提交btnSubmit.disabled = "disabled";//返回true让表单可以提交return true;}</script>
</body>
</html>

注意:通过js代码,当用户点击提交按钮后,屏蔽提交按钮使用户无法点击提交按钮或点击无效,从而实现防止表单重复提交。

三、给数据库增加唯一键约束

在创建数据库建表的时候在ID字段添加主键约束,用户名、邮箱、电话等字段加唯一性约束,以确保数据库只可以添加一条数据。数据库加唯一性约束sql:

alter table tableName_xxx add unique key uniq_xxx(field1, field2)

服务器及时捕捉插入数据异常:

try {xxxMapper.insert(user);
} catch (DuplicateKeyException e) {logger.error("user already exist");
}

注意:通过数据库加唯一键约束能有效避免数据库重复插入相同数据。但无法阻止恶意用户重复提交表单(攻击网站),服务器大量执行sql插入语句,增加服务器和数据库负荷。

四、利用Session+token防止表单重复提交(建议)

原理

服务器返回表单页面时,会先生成一个token保存于session,并把该toen传给表单页面。当表单提交时会带上token,服务器拦截器Interceptor会拦截该请求,拦截器判断session保存的token和表单提交token是否一致:若不一致或session的token为空或表单未携带token则不通过;首次提交表单时session的token与表单携带的token一致走正常流程,然后拦截器内会删除session保存的token。当再次提交表单时由于session的token为空则不通过。从而实现了防止表单重复提交。

步骤

  1. 在服务器端生成一个唯一的token(令牌),同时在当前用户的Session域中保存这个token。
  2. 将token发送到客户端的form表单中,在form表单中使用隐藏域来存储这个token,表单提交的时候连同这个token一起提交到服务器端。
  3. 在服务器端判断客户端提交上来的token与服务器端生成的token是否一致:如果不一致,那就是重复提交了,此时服务器端就可以不处理重复提交的表单;如果相同则处理表单提交,处理完后清除当前用户的Session域中存储的token。

示例

第一步:定义防止重复提交注解

在打开页面方法上,设置createToken()为true,此时拦截器会在Session中保存一个token,同时需要在页面中添加<input type="hidden" name="token" th:value="${session.token}">,保存方法需要验证重复提交的,设置removeToken为true,此时会在拦截器中验证是否重复提交

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Resubmit {/*** 创建Token* @return*/boolean createToken() default false;/*** 移除Token* @return*/boolean removeToken() default false;
}

第二步:创建拦截器

@Slf4j
public class ResubmitInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {if(handler instanceof HandlerMethod){HandlerMethod handlerMethod = (HandlerMethod) handler;Method method = handlerMethod.getMethod();Resubmit annotation = method.getAnnotation(Resubmit.class);if (annotation != null) {boolean saveSession = annotation.createToken();if (saveSession) {//在服务器端生成一个唯一的token(令牌),同时在当前用户的Session域中保存这个tokenString token = System.currentTimeMillis() + new Random().nextInt(999999999) + "";request.getSession(false).setAttribute("token", token);}boolean removeSession = annotation.removeToken();if (removeSession) {if (isRepeatSubmit(request)) {log.warn("重复提交:" + "url:" + request.getServletPath());request.setAttribute("url",request.getServletPath());response.sendRedirect(request.getContextPath()+"/resubmitError");return false;}// 处理完后清除当前用户的Session域中存储的tokenrequest.getSession(false).removeAttribute("token");}}}return true;}/*** 在服务器端判断客户端提交上来的token与服务器端生成的token是否一致* - 如果不一致,那就是重复提交了,此时服务器端就可以不处理重复提交的表单* - 如果相同则处理表单提交* @param request* @return 重复提交返回true,否则返回false*/private boolean isRepeatSubmit(HttpServletRequest request) {Object token = request.getSession(false).getAttribute("token");if(token == null){return true;}String serverToken = (String) token;if (serverToken == null) {return true;}String clientToken = request.getParameter("token");if (clientToken == null) {return true;}if (!serverToken.equals(clientToken)) {return true;}return false;}
}

第三步:配置拦截器

@Configuration
public class WegoMvcConfig implements WebMvcConfigurer {/*** 拦截器配置*/@Overridepublic void addInterceptors(InterceptorRegistry registry) {//注册防重复提交拦截器registry.addInterceptor(new ResubmitInterceptor()).addPathPatterns("/**");}
}

第四步:控制器

@Controller
public class SecurityController {/*** 打开注册页面*/@GetMapping("openRegister")@Resubmit(createToken = true)String openRegister() {return "frontend/register";}/*** 注册逻辑*/@PostMapping("/register")@Resubmit(removeToken = true)String register(UserRegisterDTO userRegisterDTO, HttpSession session, Model model) {//……}@GetMapping("/resubmitError")String error(){return "frontend/resubmitError";}
}

第五步:页面

  • register.html
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<body><form id="registerForm" class="reg" method="post" th:action="@{/user/register}"><!--防止表单重复提交--><input type="hidden" name="token" th:value="${session.token}">账户名:<input id="account" name="account" type="text"/>请设置密码:<input id="password1" name="password1" type="password"/>请确认密码:<input id="password2" name="password2" type="password"/><input  type="submit" value="注册"/></form>
</body>
</html>
  • resubmitError.html
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<body><h2>重复提交<h2>
</body>
</html>

五、使用Redis+AOP自定义切入实现 (推荐)

原理:

  1. 自定义防止重复提交标记(@AvoidRepeatableCommit)
  2. 对需要防止重复提交的Controller里的mapping方法加上该注解
  3. 新增Aspect切入点,为@AvoidRepeatableCommit加入切入点
  4. 每次提交表单时,Aspect都会保存当前key到redis(须设置过期时间)
  5. 重复提交时Aspect会判断当前redis是否有该key,若有则拦截。

示例

第一步:创建注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Resubmit {/*** 指定时间内不可重复提交,单位毫秒,默认120000毫秒*/long timeout()  default 120000 ;
}

第二步:创建增强

@Slf4j
@Aspect
@Component
public class ResubmitAspect {@Resourceprivate StringRedisTemplate stringRedisTemplate;@Around("@annotation(com.wego.common.utils.resubmit.Resubmit)")public Object around(ProceedingJoinPoint point) throws Throwable {ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest request = attributes.getRequest();Object userObj = request.getSession(false).getAttribute("user");UserSession user = null;if(userObj != null){user = (UserSession) userObj;}MethodSignature signature = (MethodSignature) point.getSignature();Method method = signature.getMethod();//目标类、方法String className = method.getDeclaringClass().getName();String methodName = method.getName();String fullMethodName = String.format("%s.%s", className, methodName);String key = String.format("%s_%d", Math.abs(user.hashCode()), Math.abs(fullMethodName.hashCode()));log.info(String.format("ipKey=%s,hashCode=%s,key=%s", fullMethodName, user, key));//通过反射技术来获取注解对象Resubmit resubmit = method.getAnnotation(Resubmit.class);long timeout = resubmit.timeout();if (timeout < 0) {//过期时间10秒timeout = 2;}//获取key键对应的值String value = stringRedisTemplate.opsForValue().get(key);if (value != null && value.length() > 0) {return  "请勿重复提交!";}//新增一个字符串类型的值,key是键,value是值。stringRedisTemplate.opsForValue().set(key, UUID.randomUUID().toString(), timeout, TimeUnit.MINUTES);//返回继续执行被拦截到的方法    return point.proceed();}
}

第三步:控制器

@RestController
public class DemoController {@Resubmit //自定义注解@GetMapping("/fun")public String fun() {System.out.println("fun");return "fun";}
}

测试

第一次请求:

再次请求:

【精品】防止表单重复提交 方法汇总相关推荐

  1. 防止表单重复提交方法

    1.背景与介绍: 平时开发的项目中可能会出现下面这些情况: 由于用户误操作,多次点击表单提交按钮. 由于网速等原因造成页面卡顿,用户重复刷新提交页面. 黑客或恶意用户使用postman等工具重复恶意提 ...

  2. 防止用户将表单重复提交的方法汇总

    表单重复提交是在多用户Web应用中最常见.带来很多麻烦的一个问题.有很多的应用场景都会遇到重复提交问题,比如: 点击提交按钮两次. 点击刷新按钮. 使用浏览器后退按钮重复之前的操作,导致重复提交表单. ...

  3. 表单重复提交的解决方法

    表单重复提交的解决方法 参考文章: (1)表单重复提交的解决方法 (2)https://www.cnblogs.com/lwj-0923/p/7367517.html 备忘一下.

  4. 5位随机数重复的概率 php_php防止表单重复提交的方法

    Token,就是令牌,最大的特点就是随机性,不可预测. Token一般用在两个地方--防止表单重复提交.anti csrf攻击(跨站点请求伪造). 两者在原理上都是通过session token来实现 ...

  5. JAVA_OA(十四)番外:JAVAWEB防止表单重复提交的方法整合(包括集群部署)

    因为自己要用,所以查找了网络上javaweb项目防止表单重复提交的方法,有些部分不太好找,所以整合后贴出来,首先是孤傲苍狼的一部分博客文章,集群部署的解决方案在后面(注意大红字) 原文出处:孤傲苍狼的 ...

  6. 防止用户将表单重复提交的方法

    2019独角兽企业重金招聘Python工程师标准>>> 表单重复提交是在多用户Web应用中最常见.带来很多麻烦的一个问题.有很多的应用场景都会遇到重复提交问题,比如: 点击提交按钮两 ...

  7. html怎么防止表单重复提交,js防止表单重复提交的解决方法

    防止表单重复提交,通常会通过attachEvent在 form的onsubmit事件中写一个方法,每次触发该事件时执行该方法,可以给form增加一个submited属性,每次判断这个属性,为 fals ...

  8. python表单防重复提交_关于PHP使用token防止表单重复提交的方法

    这篇文章主要介绍了PHP使用token防止表单重复提交的方法,通过生成一个加密后的随机数存入session的token变量,同时将该值放入表单隐藏提交,达到防止表单重复提交的功能,需要的朋友可以参考下 ...

  9. 防止表单重复提交的4种方法

    1.背景与介绍: 平时开发的项目中可能会出现下面这些情况: 由于用户误操作,多次点击表单提交按钮. 由于网速等原因造成页面卡顿,用户重复刷新提交页面. 黑客或恶意用户使用postman等工具重复恶意提 ...

最新文章

  1. JavaScript = TypeScript 类入门
  2. word文档html图片不能移动,win7系统Word中插入的图片不能移动的解决方法
  3. 判断string是否为数字
  4. 关于bjam编译自己模块出错的问题
  5. linux怎么取消挂在u盘,linux下如何挂载U盘
  6. 继承情况下构造方法的调用过程-java
  7. 【分布式系统工程实现】系统可扩展性演化
  8. MySQL数据库regdate_第十五章 MySQL 数据库
  9. 详细介绍ADMM交替方向乘子法
  10. 基于排序变换混沌置乱算法的图像加密系统
  11. 电路实习报告:简易收音机的焊接
  12. 系统之美 作者:德内拉梅多斯
  13. tinode客户端安卓版编译手账
  14. word文件打开就是只读模式,怎么取消?
  15. 【c++】设置控制台窗口字体颜色和背景色(system和SetConsoleTextAttribute函数 )
  16. UML-九种基本图形
  17. 网页字体单位px、em、%、rem、pt、vm、vh介绍
  18. tomcat 如何查看tomcat版本及位数——tomcat笔记
  19. 一文了解什么是手持气象站?手持式便携气象站分类?
  20. Android 相机服务连接失败,android-无法连接到摄像头服务

热门文章

  1. PR(precision recall curve)曲线是什么?PR曲线如何绘制?为什么Precision和Recall是矛盾体、此消彼长?为什么提出F1指标?
  2. Precision(精准率) and Recall(召回率)
  3. ARouter 源码解析(零) 基本使用
  4. R语言如何合并Excel多行的重复数据
  5. OpenLayers加载天地图
  6. 能不能推荐几个专门与外国人交友的app?这3款软件很有趣!
  7. Access-VBA
  8. python turtle 绘图小猪佩奇_python海龟作图完成小猪佩奇
  9. 【CSDN软件工程师能力认证学习精选】吐血整理!140 种 Python 标准库、第三方库和外部工具都有了
  10. 工程伦理_慕课网_第十三讲参考答案