`
sunguanxing
  • 浏览: 1109635 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

如何避免重复提交

阅读更多
这是More Java Pitfalls中文版上的一个例子。
一、何谓重复提交
   设想一个访问量非常大的订票站点,当购票者点击订票的提交按钮后,服务器的处理速度可能会迟缓,购票者迟迟见不到结果,以为自己的购买请求失败,于是一遍又一遍地点击Submit按钮,如果不加处理,这些请求都会被服务器处理,从而导致错误的结果。

二、防止重复提交

先看一下web.xml(整个例子的目录和文件请下载)
<?xml version="1.0"?> 
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/j2ee/dtds/web-app_2_3.dtd"> 

<web-app> 

  <servlet> 
    <servlet-name>ControllerServlet</servlet-name> 
    <display-name>ControllerServlet</display-name> 
    <description>ControllerServlet</description> 
    <servlet-class>org.javapitfalls.item36.ControllerServlet</servlet-class> 
        <init-param> 
                <param-name>id</param-name> 
                <param-value>id</param-value> 
        </init-param> 
  </servlet> 
   
   
  <servlet> 
        <servlet-name>form</servlet-name> 
        <jsp-file>/ticketForm.jsp</jsp-file> 
  </servlet>  

<servlet> 
        <servlet-name>success</servlet-name> 
        <jsp-file>/success.jsp</jsp-file> 
  </servlet>  
     
  <servlet> 
        <servlet-name>resubmit</servlet-name> 
        <jsp-file>/resubmitError.jsp</jsp-file> 
  </servlet>  
   
   
  <servlet-mapping> 
    <servlet-name>ControllerServlet</servlet-name> 
    <url-pattern>/ControllerServlet</url-pattern> 
  </servlet-mapping> 
   
   
</web-app> 


再看表单ticketForm.jsp
<%@ page contentType="text/html; charset=GBK" %>
<TITLE> Resubmit Demonstration </TITLE> 

<jsp:useBean id="resubmit" class="org.javapitfalls.item36.formBean" scope="request">
<jsp:setProperty name="resubmit" property="*"/>
</jsp:useBean>

<% 
//第一次提交时,在session对象中设置一个id属性,提交成功后将删除。以后的提交将检测这个属性值是否为null来判断是重复提交。
if(session.isNew()) { 
       session.setAttribute("id",session.getId()); 

%> 

<center><H1> 奥齐音乐会在线订票 </H1></center> 

<form method="get" action="/resubmit/ControllerServlet">
<center>
<table border="1" width="50%">
<tr>
<td>票的数目:</td>
<td>
<select name="numTickets">
        <option value="Please enter a Ticket #">Please enter a Ticket #</option>
        <option value="1 Ticket" <% if (resubmit.getNumTickets() == "1 Ticket") out.println("SELECTED"); %>>1 Ticket</option>
        <option value="2 Tickets" <% if (resubmit.getNumTickets() == "2 Tickets") out.println("SELECTED"); %>>2 Tickets</option>
        <option value="3 Tickets" <% if (resubmit.getNumTickets() == "3 Tickets") out.println("SELECTED"); %>>3 Tickets</option>
        <option value="4 Tickets" <% if (resubmit.getNumTickets() == "4 Tickets") out.println("SELECTED"); %>>4 Tickets</option>
        <option value="5 Tickets" <% if (resubmit.getNumTickets() == "5 Tickets") out.println("SELECTED"); %>>5 Tickets</option>
        <option value="6 Tickets" <% if (resubmit.getNumTickets() == "6 Tickets") out.println("SELECTED"); %>>6 Tickets</option>
        <option value="7 Tickets" <% if (resubmit.getNumTickets() == "7 Tickets") out.println("SELECTED"); %>>7 Tickets</option>
        <option value="8 Tickets" <% if (resubmit.getNumTickets() == "8 Tickets") out.println("SELECTED"); %>>8 Tickets</option>
        <option value="9 Tickets" <% if (resubmit.getNumTickets() == "9 Tickets") out.println("SELECTED"); %>>9 Tickets</option>
</select>
</td>
</tr>
<tr> 
<td>体育场的等级:</td>
<td>
<select name="stadiumTier">
        <option value="Please enter a Stadium Tier">Please enter a Stadium Tier</option>
        <option value="Tier A" <% if (resubmit.getStadiumTier() == "Tier A") out.println("SELECTED"); %>>Tier A</option>
        <option value="Tier B" <% if (resubmit.getStadiumTier() == "Tier B") out.println("SELECTED"); %>>Tier B</option>
        <option value="Tier C" <% if (resubmit.getStadiumTier() == "Tier C") out.println("SELECTED"); %>>Tier C</option>
        <option value="Tier D" <% if (resubmit.getStadiumTier() == "Tier D") out.println("SELECTED"); %>>Tier D</option>
        <option value="Tier E" <% if (resubmit.getStadiumTier() == "Tier E") out.println("SELECTED"); %>>Tier E</option>
</select>
</td> 
</tr>
<tr>
<td>票的价格:</td>
<td>
<select name="ticketPrice">
        <option value="Please enter a Ticket Price">Please enter a Ticket Price</option>
        <option value="$12" <% if (resubmit.getTicketPrice() == "$12") out.println("SELECTED"); %>>$12</option>
        <option value="$17" <% if (resubmit.getTicketPrice() == "$17") out.println("SELECTED"); %>>$17</option>
        <option value="$25" <% if (resubmit.getTicketPrice() == "$25") out.println("SELECTED"); %>>$25</option>
        <option value="$35" <% if (resubmit.getTicketPrice() == "$35") out.println("SELECTED"); %>>$35</option>
        <option value="$50" <% if (resubmit.getTicketPrice() == "$50") out.println("SELECTED"); %>>$50</option>
</select>
</td>
</tr>
<tr>
<td colspan="2" align="center"><INPUT TYPE="submit" VALUE="Submit"></td>
</tr>
</table>
</center>

</form> 

</BODY> 
</HTML> 

下面是处理表单的ControllerServlet.java

package org.javapitfalls.item36;

import java.io.*;
import java.util.*;
import javax.servlet.*;
import javax.servlet.http.*;
import org.javapitfalls.item36.*;

public class ControllerServlet extends HttpServlet {

        private static String SESSION_ID;

        public void destroy() {} 
  
        public void init() throws ServletException {

              SESSION_ID = getInitParameter("id"); 
        }
    
        protected void doGet(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException { 
        
             process(req, res); 
    
       } 

    protected void process(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException { 

       HttpSession session = req.getSession(false); 
       String numTickets = req.getParameter("numTickets");
       String stadiumTier = req.getParameter("stadiumTier");
       String ticketPrice = req.getParameter("ticketPrice");

       if(session == null) { 

              if( (numTickets == null) || (stadiumTier == null) || (ticketPrice == null) ) { 

                   getServletConfig().getServletContext().getNamedDispatcher("form").forward(req, res);
            
              } else { 
            
                       throw new ServletException("[form] Page Not Found"); 
            
              } 

      } else { 

         if ( (!numTickets.equals("Please enter a Ticket #")) && 
           (!stadiumTier.equals("Please enter a Stadium Tier")) && (!ticketPrice.equals("Please enter a Ticket Price")) ) {
 
          String sessionValidatorID = (String)session.getAttribute(SESSION_ID); 
          if(sessionValidatorID != null ) { //第一次提交的处理
                           session.removeAttribute(SESSION_ID); 
                           getServletConfig().getServletContext().getNamedDispatcher("success").forward(req, res);
            
                  } else { //重复提交的处理
                           getServletConfig().getServletContext().getNamedDispatcher("resubmit").forward(req, res);

                  }
         } else {

                  getServletConfig().getServletContext().getNamedDispatcher("form").forward(req, res);

         }
 
     } 
  } 
  
}

分享到:
评论

相关推荐

    j2ee服务器端避免重复提交

    下面将详细介绍如何在J2EE服务器端避免重复提交。 1. **事务管理**: J2EE服务器通常支持事务管理,通过在业务逻辑层(EJB或Spring Bean)中使用事务边界,可以确保一次只有一个提交操作被执行。例如,使用Java ...

    刷新action避免重复提交

    防止提交 在用struts2.0标签开发中,经常要使用form提交.在点击"提交"后,我们通常会弹出一个提示信息的页面,用户此时有可能会按f5刷新当前提交的action,从而将多个相同的数据保存到了后台数据库,并且...避免重复提交呢?

    修改禁止多次重复提交

    在IT行业中,尤其是在Web开发领域,防止用户多次重复提交数据是一项重要的任务,这可以避免数据库出现冗余数据,保持系统稳定。"修改禁止多次重复提交"这个话题涉及到前端交互、后端处理以及数据库操作等多个层面。...

    防止Layui form表单重复提交的实现方法

    在这个函数内部,返回false可以阻止表单的提交,即阻止表单数据被发送到服务器,从而避免重复提交。 在上述函数中,如果需要在禁止表单默认提交行为的同时,与服务器进行数据交互,则可以使用JavaScript的...

    客户端防表单重复提交和服务器端session防表单重复提交

    为了避免重复提交,可以在客户端采取一定的措施。 **JavaScript方法一:仅允许提交一次** 一种简单的实现方式是在表单提交前设置一个标记变量,如果该变量为真,则不允许再次提交。具体实现如下: ```javascript ...

    ASP.NET中防止刷新页面造成表单重复提交

    这种方式可以有效地避免重复提交,但对于需要保留用户输入数据的场景不太适用。 除此之外,本文介绍了一种更加灵活且用户体验更好的方法——利用隐藏字段(Hidden Field)配合会话状态(Session State)来判断当前...

    Struts2防止重复提交解决方案

    如果发现`token`无效,拦截器会抛出异常,导致Action不再执行,从而避免重复提交。配置`token`拦截器通常涉及到在Action类或包配置中添加`token`或`token-session`拦截器。`token-session`拦截器在处理多窗口或并发...

    防重复提交代码

    - 如果`state`已经大于`0`,并且Token验证失败,那么就不执行任何操作,这可以避免重复提交。 - 如果`state`已经大于`0`且Token验证通过,则重置Token并保存新的Token到Session中。 ##### 2. Token机制的应用 - ...

    springMVC自定义防重复提交

    当服务器接收到请求时,会检查这个标识,如果发现该标识已经存在,就拒绝此次请求,从而避免重复提交。 1. **生成Token**: - 在控制器方法执行前,可以使用`HttpServletRequest`对象的`getSession()`方法获取...

    [Jsp]防止页面表单重复提交的解决方法

    这样,即使用户刷新页面,也不会重新发送POST请求,而是进行GET请求,从而避免了重复提交。 3. **服务器端令牌验证** 在服务器生成一个唯一的令牌,将其隐藏字段添加到表单中。当表单提交时,服务器会检查令牌是否...

    Servlet、Struts、SpringMVC对于表单重复提交的解决方案

    在Servlet中,通常采用几种方法来避免重复提交: 1. **Session状态管理**:在接收到用户提交请求后,可以在Session中设置一个标志,如"formSubmitted",然后在后续的请求中检查这个标志,如果已存在则忽略请求。 2...

    8 种方案解决重复提交问题

    使用Content-MD5,可以避免重复提交,Content-MD5是指Body的MD5值,只有当Body非Form表单时才计算MD5。 ### 8. 配置注解 使用Java注解,例如@Resubmit,可以避免重复提交。 在本篇文章中,我们讨论了重复提交问题...

    利用Token机制解决重复重复提交

    - 完成对表单数据的处理后,调用`resetToken()`方法清除`HttpSession`中的Token值,避免重复提交。 #### 四、非Struts框架下的Token机制实现 对于非Struts框架的应用程序,也可以采用类似的Token机制来防止表单...

    Asp.Net中避免重复提交和弹出提示框的实例代码

    在Asp.Net开发中,确保用户只提交一次表单是...通过这种方式,我们可以确保在Asp.Net应用程序中避免重复提交,并提供一个友好的用户提示,告知他们请求正在处理。这种方法对于提升用户体验和防止数据冗余都十分有效。

    spring boot 防止重复提交实现方法详解

    Spring Boot 防止重复提交是指在用户提交表单或请求时,防止同一客户端在短时间内对同一 URL 的重复提交,从而避免服务器端的处理压力和数据的一致性问题。下面将详细介绍 Spring Boot 防止重复提交实现方法的相关...

    springboot+redis+AOP 防止表单重复提交

    在处理表单提交时,避免重复提交是一个典型的需求,以防止因网络延迟或用户误操作导致的数据冗余。本主题将深入探讨如何使用Spring Boot和Redis结合AOP(面向切面编程)来实现这一目标。 首先,让我们了解一下...

    防止刷新后退重复提交

    当尝试插入重复数据时,数据库会返回错误,从而避免重复提交。 5. **JavaScript前端控制**: 在客户端使用JavaScript检测表单是否已经被提交过。一旦提交,可以禁用提交按钮,或者显示提示信息告知用户不应再进行...

    asp.net 页面防止重复提交

    在ASP.NET开发中,页面重复提交是一个常见的问题,它可能导致数据冗余或一致性错误。防止重复提交对于确保数据的准确性和应用的稳定性至关重要。本文将深入探讨如何在ASP.NET环境中解决这个问题。 首先,理解问题的...

    解决在struts 中可以通过token 来重复提交的问题

    本文将详细介绍如何在Struts框架中利用Token机制来避免重复提交问题。 #### 一、什么是重复提交问题? 重复提交问题通常发生在用户点击“提交”按钮后网络延迟导致页面加载缓慢的情况下,用户可能会多次点击提交...

    Struts解决重复提交步骤

    特别是在一些涉及敏感操作如订单提交、支付确认等场景下,避免重复提交尤为重要。Struts框架作为Java Web应用开发的经典框架之一,提供了一套成熟的解决方案来处理这个问题——即所谓的令牌(token)机制。 #### ...

Global site tag (gtag.js) - Google Analytics