一、什么是拦截器
拦截器也是一个类
拦截器可以在Action被调用之前和之后执行代码
拦截器自由组合,增强了灵活性、扩展性,有利于系统解耦
拦截器可以拦截Action请求
在访问的Action执行之前和执行之后执行代码实现某项功能
三、开发自定义拦截器的方法
(1)实现Interceptor接口
void init():初始化拦截器所需资源
void destroy():释放在init()中分配的资源
2)TimeAction类
3)配置文件interceptor.xml
效果:
*(2)继承AbstractInterceptor抽象类
提供了init()和destroy()方法的空实现
只需要实现intercept()方法即可
welcome.jsp
show.jsp
配置文件login.xml
效果:
相似点
功能类似
原理类似
方法类似
*不同点
拦截器用于拦截对Action请求,而过滤器几乎可以过滤所有的请求
拦截器中可以获取Action的当前执行状态,而过滤器不可以
过滤器是Servlet中概念,在web.xml中配置;拦截器是Struts2/Webwork中概念,在struts.xml中配置
拦截器也是一个类
拦截器可以在Action被调用之前和之后执行代码
框架的很多核心功能是拦截器实现的
拦截器自由组合,增强了灵活性、扩展性,有利于系统解耦
拦截器可以拦截Action请求
在访问的Action执行之前和执行之后执行代码实现某项功能
三、开发自定义拦截器的方法
(1)实现Interceptor接口
void init():初始化拦截器所需资源
void destroy():释放在init()中分配的资源
String intercept(ActionInvocation ai) throws Exception:实现拦截器功能
案例1:时间拦截
1)自定义拦截器类
public class TimeInteceptor implements Interceptor {
@Override
public void destroy() {
// TODO Auto-generated method stub
System.out.println("结束拦截--->");
}
@Override
public void init() {
// TODO Auto-generated method stub
System.out.println("开始拦截--->");
}
/**
* 如同过滤器的doFilter方法 拦截一次,执行一次 执行主要的处理工作
*/
@Override
public String intercept(ActionInvocation invocation) throws Exception {
// TODO Auto-generated method stub
System.out.println("执行拦截--->");
long start = System.currentTimeMillis();
System.out.println("start=" + start);
// 进行下一个拦截器////////////////
String result = invocation.invoke();
/////////////////////////////
long end = System.currentTimeMillis();
System.out.println("minus=" + (end - start));
return result;
}
}
2)TimeAction类
public class TimeAction extends ActionSupport {
@Override
public String execute() throws Exception {
// TODO Auto-generated method stub
return SUCCESS;
}
}
3)配置文件interceptor.xml
<package name="hlx" namespace="/" extends="struts-default">
<interceptors>
<!--1)自定义拦截器 -->
<interceptor name="mytime" class="com.hlx.inter.TimeInteceptor" />
<!-- 2)声明自定义拦截器栈拦截器栈 -->
<interceptor-stack name="myStack">
<!-- 加入框架原拦截器栈 -->
<interceptor-ref name="defaultStack" />
<!--引用 自定义拦截器 -->
<interceptor-ref name="mytime" />
</interceptor-stack>
</interceptors>
<!-- 2. 配置struts启动时的拦截器栈。我们加入了自定义拦截器的栈 -->
<default-interceptor-ref name="myStack" />
<action name="time" class="com.hlx.action.TimeAction">
<!-- 引用拦截器 -->
<!-- <interceptor-ref name="mytime" /> -->
<!-- 引用拦截器栈 -->
<!-- <interceptor-ref name="myStack" /> -->
<result>/index.jsp</result>
</action>
<action name="login" class="com.hlx.action.LoginAction">
<result>/login.jsp</result>
</action>
</package>
效果:
*(2)继承AbstractInterceptor抽象类
提供了init()和destroy()方法的空实现
只需要实现intercept()方法即可
案例2: 登录拦截
login.jsp ==(y)=> welcome.jsp ==(y)==>show.jsp
welcome.jsp(interceptor) ==(a)=>无登录==>login.jsp
login.jsp
<body>
<span style="color:red;"> <s:property value="#session.msg" /></span>
<form action="login" method="post">
Name:<input type="text" name="username"><br>
Pass:<input type="password" name="password"><br>
<input type="submit" value="submit">
</form>
</body>
welcome.jsp
<body>
<h1>登录成功后显示此页面</h1>
<a href="show.action">show</a>
</body>
show.jsp
<body>
<h1>Show Main page! </h1>
</body>
登录LoginAction.java
public class LoginAction extends ActionSupport {
// 属性封装 setXXX and getXXX
private String username;
private String password;
@Override
public String execute() throws Exception {
if ("sa".equals(username) && "aaa".equals(password)) {
// 保存数据
ActionContext.getContext().getSession().put("uname", username);
ActionContext.getContext().getSession().put("upwd", password);
return SUCCESS;
}
return INPUT;
}
}
显示ShowAction.java
public class ShowAction extends ActionSupport {
@Override
public String execute() throws Exception {
// TODO Auto-generated method stub
return SUCCESS;
}
}
定义拦截器 LoginInterceptor.java
public class LoginInteceptor extends AbstractInterceptor {
@Override
public String intercept(ActionInvocation invocation) throws Exception {
// 取得请求相关的ActionContext实例
ActionContext ct = invocation.getInvocationContext();
// 获得会话对象
Map<String, Object> session = ct.getSession();
// 获取保存的会话值
String uname = (String) session.get("uname");
// 如果没有登陆,则返回重新登陆
if (uname != null && "sa".equals(uname)) {
session.put("msg", ""); //msg为空值
// 处理下一个
return invocation.invoke();
} else {
session.put("msg", "对不起,您还没有登录!");
return Action.INPUT;
}
}
}
配置文件login.xml
<package name="ha" namespace="/" extends="struts-default">
<!-- 定义一个拦截器 -->
<interceptors>
<!-- 拦截器 -->
<interceptor name="mylogin" class="com.hlx.inter.LoginInteceptor" />
<!-- 拦截器栈 -->
<interceptor-stack name="myStack">
<interceptor-ref name="defaultStack" />
<interceptor-ref name="mylogin" />
</interceptor-stack>
</interceptors>
<!-- 全局结果 -->
<global-results>
<result name="input">/login.jsp</result>
</global-results>
<action name="login" class="com.hlx.action.LoginAction">
<result>/welcome.jsp</result>
<!-- <result name="input">/login.jsp</result> -->
</action>
<action name="show" class="com.hlx.action.ShowAction">
<!-- 使用此拦截器 -->
<interceptor-ref name="myStack" />
<result>/show.jsp</result>
</action>
</package>
效果:
(1)访问welcome.jsp页面
(2)单击show超链接
说明是进行拦截,没有登录哦!
相似点
功能类似
原理类似
方法类似
*不同点
拦截器用于拦截对Action请求,而过滤器几乎可以过滤所有的请求
拦截器中可以获取Action的当前执行状态,而过滤器不可以
过滤器是Servlet中概念,在web.xml中配置;拦截器是Struts2/Webwork中概念,在struts.xml中配置