利用Session防止表单重复提交

本文详细介绍了如何通过服务器程序动态生成唯一随机标识号并存储于session中,以防止表单重复提交的问题。该机制确保了用户提交的请求安全且有效,避免了因误操作导致的数据错误。
为什么需要防止表单重复提交呢?


重复提交不仅仅是验证的问题,有时候可能会出现重复执行业务逻辑。


比如你买东西付款,如果不禁制重复提交,用户心急点两次,或者误操作点两次,

结果扣两次钱,那这个网站还不得立刻死翘翘。
重复提交更多的时候是为了安全来。
-----:
1)包含有Form表单得页面必须通过一个服务器程序动态生成,服务器程序为每次产生得页面中的form表单都分配一个唯一得随机标识号,并在form表单得一个隐藏域 保存

(2)当用户提交form得时候,负责接受这一请求得服务器程序比较form表单隐藏字段中的标识号与存贮在session中的是否相同,当下列情情况时候,服务器程序将忽略提交请求:

a.当前用户session不存在表单标识
b.用户提交得表单数据并没有标识号字段
c.存贮在当前用户的session中得标识号与表单数据中的不同


import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;


public class TokenProcessor {
private long privious;//上次生成表单标识号得时间值
private static TokenProcessor instance=new TokenProcessor();
public static String FORM_TOKEN_KEY="FORM_TOKEN_KEY";
private TokenProcessor(){

}
public static TokenProcessor getInstance(){
return instance;
}
/*
* 验证请求中得标识号是否有效,如果请求中的表单标识与当前用户session中的相同,返回结果true=
*/
public synchronized boolean isTokenValid(HttpServletRequest request){
//未避免session对象不存在时候创建Session对象
HttpSession session=request.getSession(false);
if(session==null){return false;}
String saved=(String)session.getAttribute(FORM_TOKEN_KEY);
if(saved==null){
return false;
}
String token=(String)request.getParameter(FORM_TOKEN_KEY);
if(token==null){
return false;
}
return saved.equals(token);//直接返回是否和浏览器页面是同一个token
}

/*
* 清除存储在当前用户session中的表单标识号
*/
public synchronized void reset(HttpServletRequest request){
HttpSession session=request.getSession(false);
if(session==null){
return;
}
session.removeAttribute(FORM_TOKEN_KEY);
}

/*
* 产生表单标识号并将之保存在当前用户得session中
*/

public synchronized void saveToken(HttpServletRequest request){
HttpSession session=request.getSession();
try {
byte id[]=session.getId().getBytes();
long current=System.currentTimeMillis();
if(current==privious){
current++;
}
privious=current;
byte now[]=String.valueOf(current).getBytes();
MessageDigest md=MessageDigest.getInstance("MD5");
md.update(id);
md.update(now);
String token=toHex(md.digest());
session.setAttribute(FORM_TOKEN_KEY, token);
} catch (NoSuchAlgorithmException e) {

}
}
/*
* 将一个字节数转换成十六进制得字符串
*
*/
public String toHex(byte buffer[]){
StringBuffer sb=new StringBuffer(buffer.length*2);
for (int i = 0; i < buffer.length; i++) {
sb.append(Character.forDigit((buffer[i]&0x60)>>4, 16));
sb.append(Character.forDigit(buffer[i]&0x0f, 16));
}
return sb.toString();

}
}




import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;


public class FormDoubleServlet extends HttpServlet {

//提交的时候访问的servlet
protected void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=gb2312");
PrintWriter out=response.getWriter();
TokenProcessor tokemProcessor=TokenProcessor.getInstance();
if(!tokemProcessor.isTokenValid(request)){
out.println("重复提交");
}
String p1=request.getParameter("p");
if(p1==null||p1.trim().equals("")){
out.println("请输入内容");
}else{
out.println("提交内容被处理");
tokemProcessor.reset(request);//清除session中的标识
}

}

}



<%@ page contentType="text/html; charset=GBK"%>

<%
TokenProcessor tokemProcessor=TokenProcessor.getInstance();
tokemProcessor.saveToken(request);
String token=(String)request.getSession().getAttribute(tokemProcessor.FORM_TOKEN_KEY);
%>
<html>
<head>
<title>用户登陆</title>
</head>
<body>
<form action="/testServlet" method="post">
<input name="<%=tokemProcessor.FORM_TOKEN_KEY %>" value="<%=token %>">
<input name="q"/>

<input type="submit" value="submit"/>
</form>
</body>
</html>
利用Session和token标记防止表单重复提交,通过拦截器检查token一致性,可按以下步骤实现: ### 生成Token 在服务器端生成一个唯一的Token,并将其存储在Session中。当用户访问表单页面时,将Token传递给前端页面。以下是一个简单的Java代码示例: ```java import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpSession; import java.util.UUID; public class TokenGenerator { public static String generateToken(HttpServletRequest request) { HttpSession session = request.getSession(); String token = UUID.randomUUID().toString(); session.setAttribute("formToken", token); return token; } } ``` ### 前端页面添加Token 在前端页面的表单中添加一个隐藏字段,将生成的Token放入该字段中。示例代码如下: ```html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Form</title> </head> <body> <form action="/submitForm" method="post"> <input type="hidden" name="formToken" value="${formToken}"> <!-- 其他表单字段 --> <input type="text" name="username"> <input type="password" name="password"> <input type="submit" value="Submit"> </form> </body> </html> ``` ### 实现拦截器 创建一个拦截器,在拦截器中检查请求中的Token与Session中的Token是否一致。如果一致,则允许请求继续处理;如果不一致,则拦截请求。示例代码如下: ```java import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import org.springframework.web.servlet.HandlerInterceptor; public class TokenInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { HttpSession session = request.getSession(); String sessionToken = (String) session.getAttribute("formToken"); String requestToken = request.getParameter("formToken"); if (sessionToken != null && sessionToken.equals(requestToken)) { // 验证通过,移除Session中的Token,防止重复使用 session.removeAttribute("formToken"); return true; } else { // 验证失败,拦截请求 response.sendError(HttpServletResponse.SC_BAD_REQUEST, "Invalid form token"); return false; } } } ``` ### 配置拦截器 在Spring Boot项目中,配置拦截器使其生效。示例代码如下: ```java import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; @Configuration public class WebConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new TokenInterceptor()) .addPathPatterns("/submitForm"); // 拦截的请求路径 } } ``` 通过以上步骤,就可以利用Session和token标记防止表单重复提交,通过拦截器检查token一致性。当用户第一次提交表单时,拦截器会验证Token的一致性,验证通过后移除Session中的Token,这样再次提交相同表单时,由于Token已被移除,拦截器会拦截请求,从而防止重复提交
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值