在新增和修改界面点击提交后(转发的方式跳转)
再次刷新页面,如果不做处理的话,会造成重复提交,
从而使得新增商品多次或者更改商品多次
导入servlet-api依赖和spring-webmvc依赖
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-webmvcartifactId>
<version>5.3.22version>
dependency>
<dependency>
<groupId>org.apache.tomcatgroupId>
<artifactId>servlet-apiartifactId>
<version>6.0.53version>
dependency>
package com.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/*ElementType.METHOD表示该注解只能使用在方法时*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)//运行时保留
public @interface Token {
//跳转页面时 使用
boolean save() default false;
//提交表单后 使用
boolean remove() default false;
}
package com.controller;
import com.annotation.Token;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import javax.servlet.http.HttpSession;
@Controller
@RequestMapping("emps")
public class EmpController {
@RequestMapping("add01")
@Token(remove = true)
public String add01(String ename){
System.out.println("添加员工 姓名:"+ename);
return "index";
}
@RequestMapping("toadd01")
@Token(save = true)
public String a(){
return "empadd";
}
}
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
package com.interceptor;
import com.annotation.Token;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.HandlerInterceptor;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class RepeatInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
//handler是控制器的方法
if(handler instanceof HandlerMethod){
// HandlerMethod表示控制器方法
HandlerMethod method=(HandlerMethod) handler;
//拿到方法上的token注解
Token token = method.getMethodAnnotation(Token.class);
//判断是否有使用该注解,若没有就直接放行
if (token != null) {
//判断是否为save save表示跳转页面
if (token.save()) {
request.getSession().setAttribute("token", "");
}
//判断是否为remove 表示提交表单
if (token.remove()) {
//判断是否重复提交
if(isRepeat(request)){
/*只有这一种情况才会阻止*/
request.getRequestDispatcher("/WEB-INF/repeat.jsp").forward(request, response);
return false;
}
//如果不是重复提交,就要把其删除掉
request.getSession().removeAttribute("token");
}
}
}
return true;
}
private boolean isRepeat(HttpServletRequest request) {
//判断session里面是否有值
Object token = request.getSession().getAttribute("token");
return token==null;
}
}
<mvc:interceptor>
<mvc:mapping path="/**"/>
<bean class="com.interceptor.RepeatInterceptor">bean>
mvc:interceptor>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
请勿重复提交
返回主页






当/emps/toadd方式进入添加用户界面的时候,拦截器里头会在session域中设置进去一个key为token的值
当/emps/add01方式进行表单数据提交时,会执行 if (token.remove())分支,然后此时经过验证知道token不为null,
就不属于重复提交的情况,这时需要移除掉之前设置进去的token这个key
当点击刷新的时候,就相当于再次执行/emps/add01方式,首先被拦截器拦截,发现此时token变量值==null,属于重复提交
就会转发到错误提示界面(请勿重新提交),此时就已经阻止好了重复提交了
注意:配上/**让所有页面都拦截,如果没有注解就可以放行,减低重复提交出现的可能性