AOP
Advice
:增强的具体功能;
JoinPoint
:连接点,可以简单理解为所有函数;
PointCut
:切入点,即符合切入条件的连接点;
Advisor
:advice + pointCut;
Aspect
:切面,对增强功能及切入点规则的描述。
当在Spring配置文件中添加了
<aop:aspectj-autoproxy/>
会向容器中注入AspectJAwareAdvisorAutoProxyCreator
;使用了@EnableAspectJAutoProxy
,会向容器中注入AnnotationAwareAspectJAutoProxyCreator
,这两个类实现了BeanPostProcessor
接口,在postProcessAfterInitialization
方法中获取了可以应用于该bean的增强器并生成代理对象。由于在bean的创建过程会收集所有与bean相关的处理器,因此Aop的增强功能得以实现。
以下分析基于注解实现aop功能。
(1)判断bean是否适合被代理,如果适合则执行wrapIfNecessary
:
(2)该方法中分以下两个步骤:
- 调用
getAdvicesAndAdvisorsForBean
获取适用于该bean的增强器; - 根据获取到的增强生成代理对象并返回(
由此可以得出获取到的bean对象是代理对象
)。
(3)进入获取增强的方法内部分为两个步骤:
- 获取所有的增强器;
- 从中筛选能应用于当前bean的增强。
(4)在AnnotationAwareAspectJAutoProxyCreator
中实现的查找增强器的方法中,首先调用父类的搜集方法,用于搜集基于xml文件配置的增强器;获取基于注解实现的增强器实现于this.aspectJAdvisorsBuilder.buildAspectJAdvisors()
中:
(5)遍历所有beanName,解析其对应的class,通过反射过滤获取被@Aspect
注解修饰的类
(6)遍历这些切面类,提取标注了如@Before、@After
等注解的切面方法,根据切点信息封装至InstantiationModelAwarePointcutAdvisorImpl
对象中生成增强,并在其构造函数中调用AspectJAfterAdvice
函数根据aop注解的不同生成不同类型的增强器并存入缓存(advisorsCache<beanName,List<Advisor>>
),在这些增强内部的拦截方法中通过invokeAdviceMethod
实现对增强方法(切面方法)的调用。
(7)获取到所有的增强后开始遍历,获取可以应用于该bean的增强器:
- 在遍历过程中首先根据每个增强器的PointCut相关配置获取方法匹配器 ,
- 递归获取该beanClass实现的所有接口类,通过反射获取其定义的所有方法;
- 通过方法匹配器进行匹配校验,如果匹配成功将该增强器添加至
eligibleAdvisors
,
(8)获取到适用于该bean的增强器后开始创建代理对象:
- 由于Spring中存在拦截器、增强器等多种概念,通过
advisorAdapterRegistry
统一转换为转换为Advisor
进行处理;
- 通过
proxyFactory
创建代理对象:如果目标类实现接口,并且未设置proxy-target-class=true
则使用jdk动态代理;否则采用cglib动态代理。
(9)通过Jdk动态代理创建代理对象:在实现的invoke
方法中,将所有的拦截器封装为拦截器链ReflectiveMethodInvocation
并依次调用;
(10)通过Cglib创建代理对象实现于CglibAopProxy
中:通过getCallbacks
方法将advisor封装为DynamicAdvisedInterceptor
添加至拦截链,在getProxy
方法中封装了Enhancer
对象,之后调用代理对象方法时回调DynamicAdvisedInterceptor
中的intercept
方法,该方法中依次执行所有的拦截器方法: