一,概述
1. 原理:面向切面编程
从关注点中分离出横切关注点,通俗点讲,即将解决特定领域问题的代码从业务逻辑中脱离出来。举个栗子,简单的增删改中一般有事务管理,但事务管理与业务逻辑无关,是一个特定重复的操作,这个时候,我们就可以把事务管理从增删改的业务逻辑中脱离出来。在这里,我们的关注点就是增删改等业务逻辑,横切关注点就是事务管理。
2.1 Advice通知
- 作用:
- 描述Spring AOP围绕方法调用而注入的切面行为。
- 类型(这里只列举三种)
- BeforeAdvice:在目标实例的方法(即,JoinPoint:连接点)执行前嵌入执行额外功能。
- AfterAdvice:在目标实例的方法执行后嵌入执行额外功能。
- ThrowAdvice:在目标实例的方法抛出异常时执行Advice。
2.2 Pointcut切点
决定Advice通知应该作用于哪个连接点。
2.3 Advisor通知器
用来结合Advice和Pointcut
二,Spring AOP实现的核心技术
动态代理
这里以JVM的动态代理模式举例
-
图解代理模式:
JVM动态代理.png
如图3-9,代理对象Proxy与目标对象RealSubject实现了同一个接口Subject。当调用目标对象的request()方法时,往往会被代理对象拦截。通过这种拦截, 为目标对象的方法操作做了铺垫,所以称之为代理模式。
三,建立AopProxy代理对象
1. 设计原理
ProxyFactory相关的类继承关系:
Proxy类继承关系.png
- ProxyConfig:数据基类,为子类提供配置属性
- AdvisedSupport:封装了AOP对通知和通知器的相关操作
- ProxyCreatorSupport:子类创建AOP代理对象的辅助类,继承以上提到的基类的功能实现
- AspectJProxyFactory:具体的AOP代理对象的生成。集成Spring和AspectJ。
- ProxyFactory:具体的AOP代理对象的生成。为使用Spring AOP的应用提供AOP功能的封装。需要编程式使用Spring AOP功能。
- ProxyFactoryBean:具体的AOP代理对象的生成。为使用Spring AOP的应用提供AOP功能的封装。可以在IOC容器中完成声明式配置。
2. 配置ProxyFactoryBean
<!--配置ProxyFactoryBean-->
<!--1,定义使用的通知器Advisor-->
<bean id="testAdvisor" class="com.abc.TestAdvisor"/>
<!--2,定义ProxyFactoryBean-->
<bean id="testAOP" class="org.springframework.aop.framework.ProxyFactoryBean"/>
<!--3,设定与AOP实现相关的重要属性-->
<!--3.1,proxyInterface-->
<property name="proxyInterfaces"><value>com.test.AbcInterface</value></property>
<!--3.2,target(目标对象)-->
<property name="target">
<bean class="com.abc.TestTarget"/>
</property>
<!--3.3,interceptorNames(需要定义的通知器)-->
<property name="interceptorNames">
<list><value>testAdvisor</value></list>
</property>
3. ProxyFactoryBean生成AopProxy代理对象
AopProxy的生成过程:两种方式,依赖JDK或者CGLIB提供的Proxy特性。
AopProxy的生成过程.png
- initializeAdvisorChain():初始化通知器链,通知器链中封装了一系列从配置中读取的拦截器,为代理对象的生成做好准备。
- getSingletonInstance():生成Singleton类型的Proxy
- DefaultAopProxyFactory:AopProxy接口类,AopProxy有两个子类实现,一个是JdkDynamicAopProxy,一个是CglibProxyFactory
4. JDK生成AopProxy代理对象
// JdkDynamicAopProxy生成Proxy代理对象
@Override
public Object getProxy(ClassLoader classLoader) {
if (logger.isDebugEnabled()) {
logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());
}
// 1,从advised对象中取得代理对象的代理接口配置
Class<?>[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised, true);
findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
// 2,调用JDK生成Proxy
return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
}
- newProxyInstance方法:需要指明3个参数,类装载器,代理接口,Proxy回调方法所在的对象,这个对象要实现InvocationHandler接口
- InvocationHandler接口: 反射类接口,定义了invoke方法,提供代理对象的回调入口。
5. CGLIB生成AopProxy代理对象
Cglib2AopProxy生成AopProxy代理对象
CGLIB生成Proxy代理对象.png
配置Enhancer对象,通过Enhancer对象的callback回调设置生成代理对象。其中通过设置DynamicAdvisedInterceptor拦截器来完成AOP功能的。
参考资料:Spring技术内幕(第2版) 计文柯
网友评论