拦截器是Struts2框架的核心和基础,Struts2绝大多数功能都是通过拦截器来完成的,当StrutsPrepareAndExecuteFilter拦截到用户请求后,大量拦截器会对该请求进行处理,然后才调用用户自定义的Action类中的方法进行处理请求,由此可见,拦截器是Struts2的核心所在。
Struts2框架内建了大量的拦截器,我们可以在struts-default.xml中进行查看:
那么这些内置拦截器的具体作用是什么呢?
alias |
实现在不同请求中相似参数别名的转换 |
autowiring |
这是个自动装配的拦截器,主要用于当Struts2和Spring整合时,Struts2可以使用自动装配的方式来访问Spring容器中的 Bean |
chain |
构建一个Action链,使当前Action可以访问前一个Action的属性,一般和<result type="chain" .../>一起使用 |
conversionError |
这是一个负责处理类型转换错误的拦截器,它负责将类型转换错误从ActionContext中取出,并转换成Action的FieldError错误 |
createSession |
该拦截器负责创建一个HttpSession对象,主要用于那些需要有HttpSession对象才能正常工作的拦截器中 |
debugging |
当使用Struts2的开发模式时,这个拦截器会提供更多的调试信息 |
execAndWait |
后台执行Action,负责将等待画面发送给用户 |
exception |
这个拦截器负责处理异常,它将异常映射为结果 |
fileUpload |
这个拦截器主要用于文件上传,它负责解析表单中文件域的内容 |
i18n |
这是支持国际化的拦截器,它负责把所选的语言、区域放入用户Session中 |
logger |
这是一个负责日志记录的拦截器,主要是输出Action的名字 |
model-driven |
这是一个用于模型驱动的拦截器,当某个Action类实现了ModelDriven接口时,它负责把getModel()方法的结果堆入ValueStack中 |
params |
这是最基本的一个拦截器,它负责解析HTTP请求中的参数,并将参数值设置成Action对应的属性值 |
prepare |
如果action实现了Preparable接口,将会调用该拦截器的prepare()方法。 |
scope |
这是范围转换拦截器,它可以将Action状态信息保存到HttpSession范围,或者保存到ServletContext范围内 |
servlet-config |
如果某个Action需要直接访问Servlet API,就是通过这个拦截器实现的。 |
roles |
这是一个JAAS(Java Authentication and Authorization Service,Java授权和认证服务)拦截器,只有当浏览者取得合适的授权后,才可以调用被该拦截器拦截的Action |
timer |
这个拦截器负责输出Action的执行时间,这个拦截器在分析该Action的性能瓶颈时比较有用。 |
token |
这个拦截器主要用于阻止重复提交,它检查传到Action中的token,从而防止多次提交。 |
token-session |
这个拦截器的作用与前一个基本类似,只是它把token保存在HttpSession中 |
validation |
通过执行在xxxAction-validation.xml中定义的校验器,从而完成数据校验 |
workflow |
这个拦截器负责调用Action类中的validate方法,如果校验失败,则返回input的逻辑视图 |
scoped-model-driven |
如果一个Action实现了一个ScopedModelDriven接口,该拦截器负责从指定生存范围中找出指定的Modol,并将通过setModel方法将该Model传给Action实例 |
接下来我们以token这一内置的拦截器为例来看看Struts2中拦截器的应用(token拦截器用来处理用户重复提交的页,以防止错误发生):
首先配置struts开发环境,上一篇:Struts2框架基础篇 中有介绍,这里不重复说。
写一个登录页:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@taglib uri="/struts-tags" prefix="s" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<base href="<%=basePath%>">
<title>Login</title>
</head>
<body>
<s:form action="login" method="post">
<s:token></s:token>
<s:textfield name="username" label="username"></s:textfield>
<s:password name="password" label="password"></s:password>
<s:submit value="login"></s:submit>
</s:form>
</body>
</html>
编写action:
import com.opensymphony.xwork2.ActionSupport;
public class LoginAction extends ActionSupport{
private static final long serialVersionUID = 7222063566979612268L;
private String username;
private String password;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
@Override
public String execute() throws Exception {
return SUCCESS;
}
}
接着在struts.xml中配置action及拦截器:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
"http://struts.apache.org/dtds/struts-2.3.dtd">
<struts>
<!-- 配置struts可以受理的请求扩展名 ,这是默认的-->
<constant name="struts.action.extension" value="action,,"></constant>
<package name = "hello" extends = "struts-default" namespace = "/">
<action name="login" class="org.interceptor.LoginAction">
<result>/success.jsp</result>
<result name="invalid.token">/error.jsp</result>
<interceptor-ref name="defaultStack"></interceptor-ref>
<!-- defaultStack 是系统默认的拦截器栈,会自动应用,这里手动配置的原因是当为一个action配置拦截器
时,默认的拦截器就不起作用了,所以必须显示的配置这个拦截器栈 -->
<interceptor-ref name="token"></interceptor-ref>
</action>
</package>
</struts>
最后,为了测试,编写一个成功界面以及重复提交被拦截界面:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@taglib uri="/struts-tags" prefix="s" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<base href="<%=basePath%>">
<title>welcome</title>
</head>
<body>
欢迎<s:property value="username"/>
</body>
</html>
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@taglib uri="/struts-tags" prefix="s" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<base href="<%=basePath%>">
<title>wrong</title>
</head>
<body>
请不要重复提交
</body>
</html>
测试运行:
输入用户名密码:
点击login:
刷新或者返回再次点击login,此时的操作会被拦截器拦截:
这样,用内置拦截器比较容易的就解决了表单重复提交的问题。