这是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服务器端避免重复提交。 1. **事务管理**: J2EE服务器通常支持事务管理,通过在业务逻辑层(EJB或Spring Bean)中使用事务边界,可以确保一次只有一个提交操作被执行。例如,使用Java ...
防止提交 在用struts2.0标签开发中,经常要使用form提交.在点击"提交"后,我们通常会弹出一个提示信息的页面,用户此时有可能会按f5刷新当前提交的action,从而将多个相同的数据保存到了后台数据库,并且...避免重复提交呢?
在IT行业中,尤其是在Web开发领域,防止用户多次重复提交数据是一项重要的任务,这可以避免数据库出现冗余数据,保持系统稳定。"修改禁止多次重复提交"这个话题涉及到前端交互、后端处理以及数据库操作等多个层面。...
在这个函数内部,返回false可以阻止表单的提交,即阻止表单数据被发送到服务器,从而避免重复提交。 在上述函数中,如果需要在禁止表单默认提交行为的同时,与服务器进行数据交互,则可以使用JavaScript的...
为了避免重复提交,可以在客户端采取一定的措施。 **JavaScript方法一:仅允许提交一次** 一种简单的实现方式是在表单提交前设置一个标记变量,如果该变量为真,则不允许再次提交。具体实现如下: ```javascript ...
这种方式可以有效地避免重复提交,但对于需要保留用户输入数据的场景不太适用。 除此之外,本文介绍了一种更加灵活且用户体验更好的方法——利用隐藏字段(Hidden Field)配合会话状态(Session State)来判断当前...
如果发现`token`无效,拦截器会抛出异常,导致Action不再执行,从而避免重复提交。配置`token`拦截器通常涉及到在Action类或包配置中添加`token`或`token-session`拦截器。`token-session`拦截器在处理多窗口或并发...
- 如果`state`已经大于`0`,并且Token验证失败,那么就不执行任何操作,这可以避免重复提交。 - 如果`state`已经大于`0`且Token验证通过,则重置Token并保存新的Token到Session中。 ##### 2. Token机制的应用 - ...
当服务器接收到请求时,会检查这个标识,如果发现该标识已经存在,就拒绝此次请求,从而避免重复提交。 1. **生成Token**: - 在控制器方法执行前,可以使用`HttpServletRequest`对象的`getSession()`方法获取...
这样,即使用户刷新页面,也不会重新发送POST请求,而是进行GET请求,从而避免了重复提交。 3. **服务器端令牌验证** 在服务器生成一个唯一的令牌,将其隐藏字段添加到表单中。当表单提交时,服务器会检查令牌是否...
在Servlet中,通常采用几种方法来避免重复提交: 1. **Session状态管理**:在接收到用户提交请求后,可以在Session中设置一个标志,如"formSubmitted",然后在后续的请求中检查这个标志,如果已存在则忽略请求。 2...
使用Content-MD5,可以避免重复提交,Content-MD5是指Body的MD5值,只有当Body非Form表单时才计算MD5。 ### 8. 配置注解 使用Java注解,例如@Resubmit,可以避免重复提交。 在本篇文章中,我们讨论了重复提交问题...
- 完成对表单数据的处理后,调用`resetToken()`方法清除`HttpSession`中的Token值,避免重复提交。 #### 四、非Struts框架下的Token机制实现 对于非Struts框架的应用程序,也可以采用类似的Token机制来防止表单...
在Asp.Net开发中,确保用户只提交一次表单是...通过这种方式,我们可以确保在Asp.Net应用程序中避免重复提交,并提供一个友好的用户提示,告知他们请求正在处理。这种方法对于提升用户体验和防止数据冗余都十分有效。
Spring Boot 防止重复提交是指在用户提交表单或请求时,防止同一客户端在短时间内对同一 URL 的重复提交,从而避免服务器端的处理压力和数据的一致性问题。下面将详细介绍 Spring Boot 防止重复提交实现方法的相关...
在处理表单提交时,避免重复提交是一个典型的需求,以防止因网络延迟或用户误操作导致的数据冗余。本主题将深入探讨如何使用Spring Boot和Redis结合AOP(面向切面编程)来实现这一目标。 首先,让我们了解一下...
当尝试插入重复数据时,数据库会返回错误,从而避免重复提交。 5. **JavaScript前端控制**: 在客户端使用JavaScript检测表单是否已经被提交过。一旦提交,可以禁用提交按钮,或者显示提示信息告知用户不应再进行...
在ASP.NET开发中,页面重复提交是一个常见的问题,它可能导致数据冗余或一致性错误。防止重复提交对于确保数据的准确性和应用的稳定性至关重要。本文将深入探讨如何在ASP.NET环境中解决这个问题。 首先,理解问题的...
本文将详细介绍如何在Struts框架中利用Token机制来避免重复提交问题。 #### 一、什么是重复提交问题? 重复提交问题通常发生在用户点击“提交”按钮后网络延迟导致页面加载缓慢的情况下,用户可能会多次点击提交...
特别是在一些涉及敏感操作如订单提交、支付确认等场景下,避免重复提交尤为重要。Struts框架作为Java Web应用开发的经典框架之一,提供了一套成熟的解决方案来处理这个问题——即所谓的令牌(token)机制。 #### ...