zoukankan      html  css  js  c++  java
  • JavaWeb 如何防止表单重复提交

    JavaWeb 如何防止表单重复提交 - 使用Token,令牌

    说到重复提交 ,应该想到两种场景:
    1. 在下单,或者支付 这种情况 那么不允许  刷新,不允许后退再点击提交(后退之后提交会失败,修改了也不行)。
    2. 在填写表单之后,提交完成之后,不允许 刷新,但是允许 返回之后 提交,给用户修改表单的机会。

     
     

    解决方法

    首先可以防止用户刷新,处理完成之后用Redirect的方式 跳转到success页面,这样刷新则没有用。但是返回的时候还可以提交一次缓存的数据。
     
    然后 使用令牌,在页面表单生成一个token, 这是在请求页面的时候产生的,放在隐藏域之中。然后把token存在session中。
    提交之后,判断这两个token是否一样,是一样则通过,并且清除session 中的 token,这样就能防止返回之后再次提交,因为返回的时候点击提交读取的是缓存,但是session已经没有这个token了。这适用于场景1
     
    对于场景2,直接禁止缓存,那么返回的时候一定重新请求 表单,用户可以再次填写。在用token的情况下,session中的 token总是和 隐藏域中的一致。
     
    在使用令牌的情况下,如果用户没有重新请求表单,并且恶意提交之前的Post数据,则在服务器端,session里面的token已经被清空且没有重新请求,则session的token为空不能通过。
     
    <body>
        <%
               long token=System.currentTimeMillis();    //产生时间戳的token
                session.setAttribute("token",token);    
        %>
        <form  action="isRepeat" method="post">
            <input type="text"  name="username"/>
            <input type="text"  name="password"/>
            <input type="hidden" value="<%=token %>" name="Reqtoken"/>   <!-- 作为hidden提交 -->
            <input type="submit" value="提交"/>
        </form>
    </body>


    String token = req.getParameter("Reqtoken");// 获取表单上面的时间戳T1
    		
    		HttpSession session=req.getSession();
    		String tokenInSession  =  ""+session.getAttribute("token");
    		
    		System.out.println("Session in Token: " + tokenInSession);
    		System.out.println("表单的Token:" + token+"
    ------------");
    
    		if (tokenInSession!=null && token!=null && token.equals(tokenInSession)) { 
    			resp.getWriter().println("ok ");
    			session.removeAttribute("token");
    		} else { //如果禁止缓存,返回会 重新请求,tokenInSession  是不会为空的
    			
    			System.out.println("重复提交,或者有错误");//或者有错误,直接访问servlet等
    			resp.sendRedirect("index.jsp");
    			return;
    		}
    
    		resp.sendRedirect("success.jsp"); //此句子导致刷新无效


    禁止缓存的方法,参见另一篇文章
     
     
     
     
    另一种方法: 该方法原理是 记住上一次提交的 页面token。将本次的token和上次比对,如果一样说明重复提交。该方法不需要禁用缓存。
     
     
     
                    String token = req.getParameter("Reqtoken");// 获取表单上面的时间戳
    		
    		HttpSession session=req.getSession();
    	
    		String LasttokenInSession  =  (String) session.getAttribute("Lasttoken");
    		
    		//System.out.println("------------
    Token in Session: " + tokenInSession);
    		System.out.println("表单的Token:" + token);
    		System.out.println("上一次表单的Token:" + LasttokenInSession);
    		//token!=null 是防止用户直接打开本servlet页面。LasttokenInSession是空,说明是第一次提交 ,和上一次不想等则说明不是重复提交
    		if(token!=null && (LasttokenInSession ==null || !LasttokenInSession.equals(token)) ) {
    
    			
    			session.removeAttribute("token");//Token 清空
    			
    			session.setAttribute("Lasttoken",token);// 保存上一次
    			
    		}
    		else { 
    			
    		
    				resp.getWriter().println("<h1>表单页面无效,请返回并且刷新页面</h1>");
    				resp.getWriter().println("<h1><a href=""+ "index.jsp" +"">返回并刷新</a></h1>");
    			
    			resp.getWriter().println("Do not Duplicate submit!");
    			return;
    		}
    
    		resp.sendRedirect("success.jsp"); //此句子导致刷新无效


     
  • 相关阅读:
    Android 中Base64的操作
    android 异步图片处理 工具类
    android 跨应用跳转 启动其他应用指定界面
    C++代码案例
    Android 中 Base64的操作应用
    Python基础笔记1
    [linux] 大批量删除任务
    Python基础笔记3
    [R] 如何快速生成许多差异明显的颜色?
    Python基础笔记4
  • 原文地址:https://www.cnblogs.com/slankka/p/9158541.html
Copyright © 2011-2022 走看看