最近发现很多朋友连拦截器都不知道,于是想写个BLOG总结一下。
java拦截器的基本原理其实非常简单,说白了就是动态代理类。
下面来看一个简单的例子
首先,我建立一个拦截器的类InterceptorClass,这里的before()和after()方法是以后拦截器会执行的方法
CODE清单一:
//拦截器
public class InterceptorClass {
public void before() {
System.out.println("拦截器InterceptorClass方法调用:before()!");
}
public void after() {
System.out.println("拦截器InterceptorClass方法调用:after()!");
}
}
我们模拟一个业务组件接口BusinessInterface,和一个业务组件实现类BusinessClass
CODE清单二:
/**
* Created by IntelliJ IDEA.
* User: Ming
* Date: 2010-9-7
* Time: 8:45:26
*/
//业务组件接口
public interface BusinessInterface {
public void doSomething();
}
CODE清单三:
/**
* Created by IntelliJ IDEA.
* User: Ming
* Date: 2010-9-7
* Time: 8:46:27
*/
//业务组件实现类
public class BusinessClass implements BusinessInterface {
public void doSomething() {
System.out.println("业务组件BusinessClass方法调用:doSomething()");
}
}
然后,创建一个动态代理类DynamicProxyHandler,这个类是集成InvocationHandler接口的,动态类的原理实际上是使得当你执行一个动态方
法的时候,他可以把这个动态方法dispatch到这个动态类上来。这样,你就可以在这个方法的前后嵌入自己的一些方法。
CODE清单四:
/**
* Created by IntelliJ IDEA.
* User: Ming
* Date: 2010-9-7
* Time: 8:50:35
*/
/**
* 动态代理类,实现InvocationHandler接口.
* 包含了业务对象绑定动态代理类的处理
*/
public class DynamicProxyHandler implements InvocationHandler {
private Object business;//被代理对象
private InterceptorClass inceptor = new InterceptorClass();//拦截器
public DynamicProxyHandler(Object business) {
this.business = business;
}
// 代理要调用的方法,并在方法调用前后调用连接器的方法
/**
* @param proxy 代理类对象
* @param method 被代理的接口方法
* @param args 被代理接口方法的参数
* @return
* @throws Throwable
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = null;
inceptor.before();
result = method.invoke(business, args);
inceptor.after();
return result;
}
OK,我们来写个类测试一下
CODE清单五:
public static void main(String[] args) {
//生成待测试的业务组件对象
BusinessInterface business = new BusinessClass();
//生成动态代理类实例
DynamicProxyHandler handler = new DynamicProxyHandler(business);
BusinessInterface businessProxy = (BusinessInterface) Proxy.newProxyInstance(
//被代理类的ClassLoader
business.getClass().getClassLoader(),
//要被代理的接口,本方法返回对象会自动声称实现了这些接口
business.getClass().getInterfaces(),
//代理处理器对象
handler);
//用动态代理类调用方法
businessProxy.doSomething();
}
来看看结果:
近期struts2很流行,而且拦截器是struts2里面一个比较好的功能,下面举个例子说明一下拦截器在struts2中的用法。
struts2对拦截器实现做了一个封装,使得我们在实现的时候比较简单。
首先我们要建一个拦截器类
CODE清单六:
public class AuthorizationInterceptor extends AbstractInterceptor {
@Override
public String intercept(ActionInvocation invocation) throws Exception {
Map session = invocation.getInvocationContext().getSession();
String userName = (String) session.get("userName");
if ( userName != null && userName.equals("test")) {
System.out.println("拦截器:合法用户登录---");
return invocation.invoke();
}
else
{
System.out.println("拦截器:用户未登录---");
return Action.LOGIN;
}
}
}
这个类是必须要继承struts2包中提供的AbstractInterceptor类,这个类有一个抽象方法intercept,这个方法是必须要实现的。
那么经理在这个拦截器里面写了一个简单的实现,对url用户合法性做了一个限制。
接下来比较关键的是过滤器在struts2中的配置,先看看代码
CODE清单七:
<package name="system" extends="struts-default">
<interceptors>
<!-- 定义权限控制拦截器 -->
<interceptor name="authority"
class="com.sharesin.biz.common.intercepts.struts2.AuthorizationInterceptor"/>
<!-- 定义一个包含权限控制的拦截器栈 -->
<interceptor-stack name="mystack">
<interceptor-ref name="defaultStack"></interceptor-ref>
<interceptor-ref name="authority"></interceptor-ref>
</interceptor-stack>
</interceptors>
<!--定义默认拦截器 -->
<default-interceptor-ref name="mystack" />
<!--定义全局处理结果 -->
<global-results>
<result name="login">index.jsp</result>
</global-results>
<action name="login_*" class="com.sharesin.biz.web.system.LoginAction" method="{1}">
<result name="success">system/homepage.jsp</result>
</action>
</package>
在interceptors节点里,我们可以定义多个拦截器,这里的名为authority的只是其中的一个。struts2的拦截器栈我是先执行struts2默认的拦
截器defaultStack,然后再执行我的。然后只需要用default-interceptor-ref标签设置好这个system包中的默认拦截器为这个拦截器就OK了。
struts2中引入了package这个概念,我觉得十分实用,当然这对struts2拦截器也是个实惠,我们可以根据不同的action来分包和不同的拦截器
ok,来运行测试一下。