当前位置: 移动技术网 > IT编程>开发语言>Java > Java表单重复提交的避免方法

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

2019年07月22日  | 移动技术网IT编程  | 我要评论

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

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

造成重复提交的现象:

  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;
  }
}

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

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

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网