随笔-86  评论-767  文章-3  trackbacks-3

本文为本人在dev2dev回答网友提问所作:
原始链接: http://dev2dev.bea.com.cn/bbs/thread.jspa?forumID=121&threadID=22185&tstart=0
利用Token解决重复重复提交:
Struts利用同步令牌(Token)的方式来解决Web应用中重复提交的问题,其机制是在form表单中增加一个隐藏的域,保存当前令牌值,然后在程序中判断此令牌值是否合法.
org.apache.struts.action.Action类提供了相关操作Token的方法:
1、isTokenValie方法:
判断存储在当前用户会话中的令牌值和请求参数中的令牌值是否匹配.如果匹配,返回true,反之返回false.只要符合下列情况之一的,就会返回false:
不存在HttpSession对象;
在session范围内没有保存令牌值;
在请求参数中没有令牌值;
存储在当前用户session范围内的令牌值和请求参数中的令牌值不匹配.
2、resetToken方法:
从当前session范围内删除令牌属性.
3、saveToken方法:
创建一个新的令牌,并把它保存在当前session范围内.如果HttpSession不存在,就首先创建一个HttpSession对象.

如何利用上述方法应用令牌机制解决重复提交问题:
以用户注册为例:
在用户请求newUser.jsp之前,首先把请求转发到PrepareAction,PrepareAction 调用saveToken方法,创建一个新的令牌,并将令牌值保存在当前HttpSession中(新创建的),PrepareAction接着把请求转发给newUser.jsp.

newUser.jsp中的<html:form>标签自动判断在session范围内是否存在Token,如果存在,就自动在表单中生成一个包含Token信息的隐藏字段,例如:
<input type="hidden" name="org.apache.struts.taglib.html.TOKEN" value="....">

在用户提交表单后,由InsertUserAction处理请求.在InsertUserAction中,调用isTokenValid方法,判断当前用户会话中的令牌值和请求参数中的令牌值是否匹配.如果匹配,就调用resetToken方法,删除Token,然后执行插入数据操作.如果不匹配,返回相关错误提示,进行相关操作.

OK.酱紫就可以有效放置重复提交了.

如何在不使用Struts的前提下利用令牌机制解决重复提交问题:
Struts的令牌机制有几个要点可以让我们在普通的JSP/Servlet中解决重复提交问题.
1、提供几个操作Token的相关方法:
关键有三个:
resetToken(HttpServletRequest request)-->重置令牌值.
saveToken(HttpServletRequest request)-->保存令牌值.
isTokenValid(HttpServletRequest request)-->检测令牌是否合法.

2、在form表单中增加隐藏域,保存当前令牌值.

3、在执行持久性数据操作之前调用相关方法判断当前令牌是否合法,之后在进行相关操作.

方法是相同的,只是实现的方式不太一样.

truts1.1 API关于几个Token操作方法的说明:

protected  boolean isTokenValid(javax.servlet.http.HttpServletRequest request)
          Return true if there is a transaction token stored in the user's current session, and the value submitted as a request parameter with this action matches it.

protected  boolean isTokenValid(javax.servlet.http.HttpServletRequest request, boolean reset)
          Return true if there is a transaction token stored in the user's current session, and the value submitted as a request parameter with this action matches it.

protected  void resetToken(javax.servlet.http.HttpServletRequest request)
          Reset the saved transaction token in the user's session.

protected  void saveToken(javax.servlet.http.HttpServletRequest request)
          Save a new transaction token in the user's current session, creating a new session if necessary.

posted on 2005-04-18 10:08 eamoi 阅读(7003) 评论(7)  编辑  收藏 所属分类: Java

评论:
# re: 利用Token机制解决重复重复提交问题 2005-05-09 15:04 | Jim
hi,
看了上面的文章。
我有一个疑问,在workshop中,有没有哪个tag可以自动生成与token相关的隐藏字段呢?  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题 2005-05-09 15:21 | Jim
Hi,你的文章中提到:
“newUser.jsp中的<html:form>标签自动判断在session范围内是否存在Token,如果存在,就自动在表单中生成一个包含Token信息的隐藏字段,例如:
<input type="hidden" name="org.apache.struts.taglib.html.TOKEN" value="...."> ”

《----请问:你所说的自动生成,是直接使用tag吗? <html:form> 可以吗?怎么那个标签好像是生成一个 table呢?

希望能够看到你写的一个完整的例子,谢谢
  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题 2005-05-09 17:56 | Jim
我已经搞定了这个问题,并且使用workshop做了一个具体的例子。在dev2dev上可以找到。 wujiboy  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题 2006-11-07 11:43 | asean11
好文章,有实例吗,发给我运行看看效果,谢谢
QQ67705705

邮箱:zhyf08@163.com  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题 2006-11-07 11:45 | asean11
@Jim
  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题 2007-07-31 10:59 | like
有点迷糊  回复  更多评论
  
# re: 利用Token机制解决重复重复提交问题[未登录] 2008-08-16 14:11 | liu
userView.jsp->PrepareAction(Forward)->userView.jsp->InsertUserAction  回复  更多评论
  

只有注册用户登录后才能发表评论。


网站导航:
博客园   IT新闻   Chat2DB   C++博客   博问