欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Java表单重复提交的避免方法

程序员文章站 2024-03-31 19:55:22
表单的重复提交: 没有完整的进行一次,先请求表单页面->再提交表单过程而完成数据提交 造成的根本原因: 没有完整的进行一次,先请求表单页面->再提交表单过程....

表单的重复提交: 没有完整的进行一次,先请求表单页面->再提交表单过程而完成数据提交

造成的根本原因: 没有完整的进行一次,先请求表单页面->再提交表单过程.

造成重复提交的现象:

  1. 由于服务器缓慢或网络延迟的原因,重复点击提交按钮.
  2. 已经提交成功,刷新成功页面(forward)(请求转发).
  3. 已经提交成功,通过回退,再次点击提交按钮

注意:回退后,刷新表单页面,重新再提交,这时,不是重复提交,而是发送新的请求,在firefox下,重复提交到同一个地址的操作无效.

案例:

@webservlet("/trans")
public class transferservlet extends httpservlet{
  private static final long serialversionuid = 1l;
  
  protected void service(httpservletrequest req, httpservletresponse resp)
      throws servletexception, ioexception {
    req.setcharacterencoding("utf-8");
    resp.setcontenttype("text/html;charset=utf-8");
    printwriter out = resp.getwriter();
    string money = req.getparameter("money");
    //通过睡眠,模拟网络延迟
    try {
      thread.sleep(3000);
    } catch (interruptedexception e) {
      e.printstacktrace();
    }
    system.out.println("转出金额:"+money);
    out.print("转出金额:"+money);  
  }
}

在狂点之下,会发现,jsp页面不会有变化,但是通过后台的打印输出可以看到,会不停地输出,说明一直在执行

解决方法:

保证提交保证之前,就必须先请求表单界面,原理和验证码一样.----令牌机制

在第一次请求的时候,请求到表单界面时,创建一个令牌,当点击转账,发送请求的时候,带上这个令牌,发送到下一个界面,在servlet中对这个令牌进行验证,这个令牌在session中一份,在表单中一份,相等说明表单正确,并且把这个session中的令牌销毁掉.

在jsp页面中的代码

<%
      //创建令牌
      string token = java.util.uuid.randomuuid().tostring();
      //存在session中一份,以后做判断
      session.setattribute("token_in_session", token);
    %>
    
    <h3>转账界面</h3>
    <form action="/trans" method="post">
      <input type="hidden" name="token" value="<%=token%>"/>
      转账金额:<input type="text" name="money" min="1" required /><br/>
      <input type="submit" value="转账" />
    </form>

在transferservlet中的代码

@webservlet("/trans")
public class transferservlet extends httpservlet{
  private static final long serialversionuid = 1l;
  
  protected void service(httpservletrequest req, httpservletresponse resp)
      throws servletexception, ioexception {
    req.setcharacterencoding("utf-8");
    resp.setcontenttype("text/html;charset=utf-8");
    printwriter out = resp.getwriter();
    //获取表单中的token值
    string token = req.getparameter("token");
    //获取session中的token值
    string sessiontoken = (string) req.getsession().getattribute("token_in_session");
    //session中的token容易为空,因为session中的token是需要被销毁的
    
    if (token.equals(sessiontoken)) {
      //说明令牌相同
      req.getsession().removeattribute("token_in_session");
      string money = req.getparameter("money");
      system.out.println("转出金额:"+money);
      out.print("转出金额:"+money);
      //最后销毁session中的令牌
    }
    //如果令牌不同说明就是重复提交,不能提交
  }
}

然后呢,为了不想在jsp文件中出现java代码,把令牌的创建并跳转放在另一个servlet中

@webservlet("/transfer")
public class copyoftransferservlet extends httpservlet{
  private static final long serialversionuid = 1l;
  
  protected void service(httpservletrequest req, httpservletresponse resp)
      throws servletexception, ioexception {
    //创建令牌,并跳转到submit.jsp
    string token = uuid.randomuuid().tostring();
    system.out.println(token);
    req.getsession().setattribute("token_in_session", token);
    req.setattribute("token", token);
    req.getrequestdispatcher("/views/repeatsubmit/submit.jsp").forward(req, resp);
    
  }
}

此时jsp文件中就是这个样子

<h3>转账界面</h3>
    <form action="/trans" method="post">
      <input type="hidden" name="token" value="${token }"/>
      转账金额:<input type="text" name="money" min="1" required /><br/>
      <input type="submit" value="转账" />
    </form>

比上面的干净了很多,更加整齐,但是还是感觉不好,因为如果在其他地方需要用到,还需要在创建令牌,校验令牌,删除令牌,因此就抽取出来做一个工具类

tokenutil.java

//令牌的工具类
//创建令牌
//校验令牌
//销毁令牌
public class tokenutil {
  private final static string token_in_session = "token_in_session";
  public static void savatoken(httpservletrequest req) {
    string token = uuid.randomuuid().tostring();
    system.out.println(token);
    req.getsession().setattribute(token_in_session, token);
    req.setattribute("token", token);
  }

  public static boolean validatetoken(httpservletrequest req,
      string tokeninrequest) {
    //获取session中的token值
    string sessiontoken = (string) req.getsession().getattribute(
        token_in_session);
    if (tokeninrequest.equals(sessiontoken)) {
      req.getsession().removeattribute(token_in_session);
      return true;
    }
    return false;
  }
}

这样就好了,使用者只需要调用这个工具类就好了,不需要再去写创建令牌等一系列操作。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。