Spring3核心技术之AOP动态代理

在进行AOP开发前,先熟悉几个概念
连接点(Jointpoint):表示需要在程序中插入横切关注点的扩展点,连接点可能是类初始化、方法执行、方法调用、字段调用或处理异常等等,Spring只支持方法执行连接点,在AOP中表示为“在哪里干”;
切入点(Pointcut):选择一组相关连接点的模式,即可以认为连接点的集合,Spring支持perl5正则表达式和AspectJ切入点模式,Spring默认使用AspectJ语法,在AOP中表示为“在哪里干的集合”;
通知(Advice):在连接点上执行的行为,通知提供了在AOP中需要在切入点所选择的连接点处进行扩展现有行为的手段;包括前置通知(before advice)、后置通知(after advice)、环绕通知(around advice),在Spring中通过代理模式实现AOP,并通过拦截器模式以环绕连接点的拦截器链织入通知;在AOP中表示为“干什么”;
方面/切面(Aspect):横切关注点的模块化,比如上边提到的日志组件。可以认为是通知、引入和切入点的组合;在Spring中可以使用Schema和@AspectJ方式进行组织实现;在AOP中表示为“在哪干和干什么集合”;
引入(inter-type declaration):也称为内部类型声明,为已有的类添加额外新的字段或方法,Spring允许引入新的接口(必须对应一个实现)到所有被代理对象(目标对象), 在AOP中表示为“干什么(引入什么)”;
目标对象(Target Object):需要被织入横切关注点的对象,即该对象是切入点选择的对象,需要被通知的对象,从而也可称为“被通知对象”;由于Spring AOP 通过代理模式实现,从而这个对象永远是被代理对象,在AOP中表示为“对谁干”;
AOP代理(AOP Proxy):AOP框架使用代理模式创建的对象,从而实现在连接点处插入通知(即应用切面),就是通过代理来对目标对象应用切面。在Spring中,AOP代理可以用JDK动态代理或CGLIB代理实现,而通过拦截器模型应用切面。
织入(Weaving):织入是一个过程,是将切面应用到目标对象从而创建出AOP代理对象的过程,织入可以在编译期、类装载期、运行期进行。

在AOP中,通过切入点选择目标对象的连接点,然后在目标对象的相应连接点处织入通知,而切入点和通知就是切面(横切关注点),而在目标对象连接点处应用切面的实现方式是通过AOP代理对象

AOP代理
AOP代理就是AOP框架通过代理模式创建的对象,Spring使用JDK动态代理或CGLIB代理来实现,Spring默认使用JDK动态代理来实现,但是只有实现接口的类的对象可被代理,如果被代理的对象实现不是接口将默认使用CGLIB代理,不过CGLIB代理当然也可应用到接口。

AOP代理的目的就是将切面织入到目标对象。

Spring AOP通过代理模式实现,目前支持两种代理:JDK动态代理、CGLIB代理来创建AOP代理,Spring建议优先使用JDK动态代理。要用到Spring AOP必须加入这些jar包:
25003645_FHca.jpg
JDK动态代理:使用java.lang.reflect.Proxy动态代理实现,即提取目标对象的接口,然后对接口创建AOP代理。
CGLIB代理:CGLIB代理不仅能进行接口代理,也能进行类代理,CGLIB代理需要注意以下问题:
       不能通知final方法,因为final方法不能被覆盖(CGLIB通过生成子类来创建代理)。
       会产生两次构造器调用,第一次是目标类的构造器调用,第二次是CGLIB生成的代理类的构造器调用。如果需要CGLIB代理方法,请确保两次构造器调用不影响应用。

Spring AOP默认首先使用JDK动态代理来代理目标对象,如果目标对象没有实现任何接口将使用CGLIB代理,如果需要强制使用CGLIB代理,请使用如下方式指定:
对于xml配置切面使用如下方式来指定使用CGLIB代理:

<aop:config proxy-target-class="true">  
</aop:config>

对于注解配置切面使用如下方式来指定使用CGLIB代理:
<aop:aspectj-autoproxy proxy-target-class="true"/>

下面来模拟一下JDK动态代理:
JDK动态代理首先要实现InvocationHandler接口,然后调用Proxy.newProxyInstance()静态方法返回目标对象的代理对象

/**
 * @author Chou
 * 创建动态代理对象有两种方法:
 * 1.像本类一样使用JDK的Proxy实现动态代理(需要目标对象的类必须有实现的接口)
 * 2.使用cglib这个jar包(目标对象的类不需要实现接口)
 * 注:Spring框架都用到了这两种方式,如果有实现接口,就第一种,没有实现就用第二种创建代理对象
 */
public class JDKProxyFactory implements InvocationHandler{
	
	private Object target;//需要代理的目标对象
	
	/**
	 * 通过Proxy.newProxyInstance()返回target这个对象的代理对象。
	 * 方法中需要target对象的类加载器target.getClass().getClassLoader()
	 * 还有target对象实现的所有的接口,target.getClass().getInterfaces()
	 * 既然target实现了这么多接口,那么返回的target的代理对象肯定也要实现这些接口
	 * this这个对象必须是实现了InvocationHandler接口的对象
	 */
	public Object createProxyIntance(Object target) {//target需要代理的对象
		this.target = target;
		return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
	}
	
	
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		Interceptor.prepareDomain();//加入的拦截方法
		return method.invoke(target, args);
	}

}

public class MyBeanB implements MyBean{
	public void domain() {
		System.out.println("MyBeanB is executing...");
	}
}

//拦截方法
public class Interceptor {
	public static void prepareDomain() {
		System.out.println("This is prepareDomain....");
	}
}

//main方法....
JDKProxyFactory aop = new JDKProxyFactory();
String[] configs = new String[] {"applicationContext.xml"};
ApplicationContext cxt = new ClassPathXmlApplicationContext(configs);
MyBean proxyObj = (MyBean)aop.createProxyIntance(cxt.getBean("beanB",MyBeanB.class));
proxyObj.domain();

下面来模拟一下CGLIB动态代理:
这种代理目标对象的方式必须加入CGlib的jar包,并且实现MethodInterceptor接口

public class CGlibProxyFactory implements MethodInterceptor{
	
	private Object target;//需要代理的目标对象
	
	public Object createProxyIntance(Object target) {
		this.target = target;
		Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(this.target.getClass());
		enhancer.setCallback(this);
		return enhancer.create();
	}

	@Override
	public Object intercept(Object object, Method method, Object[] args,
			MethodProxy methodProxy) throws Throwable {
		Interceptor.prepareDomain();
		return method.invoke(target, args);
	}
}

public class MyBeanB{
	public void domain() {
		System.out.println("MyBeanB is executing...");
	}
}

//main方法...
String[] configs = new String[] {"applicationContext.xml"};
ApplicationContext cxt = new ClassPathXmlApplicationContext(configs);
CGlibProxyFactory aop = new CGlibProxyFactory();
MyBeanB proxyObj = (MyBeanB) aop.createProxyIntance(cxt.getBean("beanB",MyBeanB.class));
proxyObj.domain();

在spring中获取代理对象代理的目标对象工具类:
http://jinnianshilongnian.iteye.com/blog/1613222

转载于:https://my.oschina.net/easyean/blog/535117

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值