`

看透Spring AOP

阅读更多
工作原因需要用到一个代理,首选Spring Aop ,很久没用了,今天翻了下文档看了下源码,发现看文档容易晕,看源码结构清晰明了,Advice,PointCut,Advisor泾渭分明,ProxyFactoryBean总控。
Target:主角,目的就是生成target的代理对象。
Advice:抛弃概念简单理解就是你要插入target的执行代码。
PointCut:就是你要在target文件的哪个方法哪个位置插入,之前还是之后。
Advisor:用来整合Advice与PointCut。
Spring Aop复杂在对于前四者再加上Interceptor的封装,对这四者的装饰封装衍生出了好多类,而实际上就是Proxy.newProxyInstance()罢了(对于非接口当然是耳熟能详的cglib),接下来用JDK动态代理来模拟一下,不想写那么多类了,就Advice植入Target吧,其他硬编码了。
advice接口及两个实现:
package cn.easyweb.aop;

import java.lang.reflect.InvocationHandler;

public interface Advice extends InvocationHandler{
	public Object getTarget();	
}

package cn.easyweb.aop;

import java.lang.reflect.Method;

public class BeforeAdvice implements Advice{
	
	public Object target;
	
	public BeforeAdvice(Object target){
		this.target = target;
	}
	
	
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("before....");
		method.invoke(target, args);
		return null;
	}


	public Object getTarget() {
		return target;
	}

}
package cn.easyweb.aop;

import java.lang.reflect.Method;

public class AfterAdvice implements Advice{
	
	public Object target;
	
	public AfterAdvice(Object target){
		this.target = target;
	}

	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		method.invoke(target, args);
		System.out.println("after.....");
		return null;
	}

	public Object getTarget() {
		return target;
	}

}

主控类EasyProxy :
package cn.easyweb.aop;


public class EasyProxy {
	
	public Advice advice;
	
	public Object createProxy(Advice advice){
		this.advice = advice;
		return new AopFactory().getObject(advice.getTarget(), advice);
	}

}

代理工厂:
package cn.easyweb.aop;

import java.lang.reflect.Proxy;

public class AopFactory {
	
	public Object getObject(Object target,Advice advice){
		return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), advice);
	}

}

需要代理的接口及实现类:
package cn.easyweb.aop;

public interface SourceAssaign {
	
	public void assaign();

}
package cn.easyweb.aop;


public class SimcardAssaign implements SourceAssaign {

	public void assaign() {
		System.out.println("simcard assaign");
	}

}


测试类:
package cn.easyweb.aop;


public class TestAOP {
	
	public static void main(String[] args) {
		EasyProxy proxy = new EasyProxy();
		Object obj =  proxy.createProxy(new BeforeAdvice(new SimcardAssaign()));
		SourceAssaign assaign = (SourceAssaign) proxy.createProxy(new AfterAdvice(obj));
		assaign.assaign();
		
	}
}

Advice为继承InvocationHandler的接口,抽象个接口用来实现代理链,target简单起见聚合进advice。

aop的应用非常广,除了平时经常见到的事物日志啊这些乱七八糟的控制 以外,前两天看连接池时候突然发现c3p0的也用到了动态代理,连接池产生的connection全部用动态代理产生,好处是每次调用connection.close()时,防止被真正关闭,而是让其回归连接池,当然使用spring的人估计是很难发现这点,这也是用惯了封装框架的弊端啊。

另外个人觉得spring aop之所以强大的原因还是有IOC在支持,除去IOC spring aop无非就是对几个代理点封装的好而已,单独用AspectJ都比这强,但是一加入IOC就真正做到了热插拔,侵入性一下子消失了,所以spring的核心个人觉得还应该是IOC,而不是IOC+AOP。

下次再写一篇spring jdbc的,spring jdbc的设计相当完美,用模版和回调解决jdbc的冗余代码,Rod Johnson太有才了。。。。。。
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics