J2SE Java基础 J2EE 设计模式 struts hibernate spring freemarker ibatis xml xslt Javascript mysql Linux 系统架构
Java高级 webwork webservice 网页制作 oracle sql server 数据库基础 软件工程 网站建设 SEO 黑客基础 名人堂 Ruby
首页 > 技术文档 > Java > 开源相关 > struts > 利用Token防止重复提交(Struts框架)

利用Token防止重复提交(Struts框架)

作者: unknow 来源: web  标签:struts token 重复提交 (English)
如果用户对一个html表单多次提交,web应用应该能够判断用户的重复提交行为,并作出相应的处理。

最常见的是新增一条数据,用户已经提交表单并且服务器端已经完成新增成功。此时用户可能有两个误操作:
1.用户通过浏览器的后退功能,返回到录入页面,重复提交(此时浏览器提供回退功能基本上是个邪恶行为)
2.刷新该页面(因为新增成功的提示页面通常是通过请求转发(forward)过来的,所以此操作实际效果通常等同于1)

这样造成的可能结果有:
1.若程序级别和数据库级别限制了重复记录,会提示类似于“xxx字段已存在,请修改后重新保存”的信息
2.若没有此限制,服务器端会再插入一条数据,而这通常不是用户想要的

误操作2和可能结果2的结合就达成了与用户意图相背的结果:服务器端不停地在增加重复记录,用户认为自己只不过是刷新该提示信息页面。

通用的解决思路是:
用户请求录入页面,这个与服务器建立的一次连接过程中,在服务器端①【生成一个session标识,同时返回到客户端一个与此匹配的hidden域】。用户提交了此页面,服务器端首先②【判断此hidden域与session标识是否匹配】,若不匹配,终止保存操作,提示同一表单不能提交两次,同时①【新建一个session标识和hidden域】,返回录入页面;若匹配,执行插入保存操作,同时③【清空(重置reset)session标识】。

Struts正在基于这样的思路在org.apache.struts.action.Action类中提供了内置支持方法:
java代码:

protected void saveToken(HttpServletRequest request) 配合标签对应于①
protected boolean isTokenValid(HttpServletRequest request) 对应于②
protected void resetToken(HttpServletRequest request) 对应于③

这样我们在写程序的时候,结合Struts的html标签,只要
1.在forward到insert.jsp页面前加一个action执行saveToken(request)操作,或干脆在insert.jsp中写
2.保存前加个判断操作isTokenValid(request)
3.若isTokenValid(request)返回false,执行saveToken(request)操作,返回错误提示页面;true则执行resetToken(request)操作,然后进行实际的保存操作

我的代码


// 进入创建前的Action
public final class EnterNewProjectPageAction extends Action {
        public ActionForward execute(ActionMapping mapping, ActionForm form,
                        HttpServletRequest request, HttpServletResponse response)
                        throws Exception {       
               
                saveToken(request);
               
                return (mapping.findForward("createproject_jsp"));
        }
}

// 创建Action
public final class CreateProjectAction extends Action {
        public ActionForward execute(ActionMapping mapping, ActionForm form,
                        HttpServletRequest request, HttpServletResponse response)
                        throws Exception {       

               
                if(isTokenValid(request)){
                        resetToken(request);
                       
                        CreateProjectForm createProjectForm=(CreateProjectForm)form;       
                        Project newProject=createProjectForm.getProject();
                       
                        newProject.setCreateTime(Util.getCurrTime());
       
                        ProjectService.create(newProject);       
                       
                }
                else{
                        saveToken(request);
                        System.out.println("重复提交");
                }               
       
                ProjectUtil.setupPages(0,request);
                return (mapping.findForward("projects_jsp"));
        }
}

相关文章
(2007年04月17日)Struts - 标签库
(2007年06月13日)常用的Struts 2.0的标志(Tag)介绍
(2007年06月06日)struts2的struts.properties详解
(2007年06月13日)为Struts 2.0做好准备
(2007年10月08日)struts2的struts.properties程序解析

评论列表

发表评论

赞助商