(一)注解方式注册流程原理
是在配置类上配置@EnableAspectJAutoProxy的
需要注意的是@EnableAspectJAutoProxy 需要被@ComponentScan扫描到才行,不然没有作用.
被@ComponentScan扫描到之后,会加载这个注解里面的@Import注解标注的AspectJAutoProxyRegistrar类(ImportBeanDefinitionRegistrar的实现类)
AspectJAutoProxyRegistrar被导入进来了,Spring在容器启动的时候ConfigurationClassBeanDefinitionReader#loadBeanDefinitionsFromRegistrars会调用到registerBeanDefinitions,
就会执行AspectJAutoProxyRegistrar#registerBeanDefinitions 内部逻辑是
—————————-
调用
AopConfigUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary(org.springframework.beans.factory.support.BeanDefinitionRegistry)调用
AopConfigUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary(org.springframework.beans.factory.support.BeanDefinitionRegistry, java.lang.Object)调用
AopConfigUtils#registerOrEscalateApcAsRequired给
AnnotationAwareAspectJAutoProxyCreator 变成BeanDefinition对象,放入BeanDefinitionRegistry里面
—————————————
然后AspectJAutoProxyRegistrar#registerBeanDefinitions方法接着执行解析
@EnableAspectJAutoProxy上面的配置属性,比如说指定JDK还是CGLib动态代理等等配置属性,解析完了保存到BeanDefinitionRegistry里面的BeanDefinition
最后通过BeanDefinitionRegistry给BeanDefinition对象注册到Spring容器里面.
AspectJAutoProxyRegistrar
class AspectJAutoProxyRegistrar implements ImportBeanDefinitionRegistrar {
/**
* Register, escalate, and configure the AspectJ auto proxy creator based on the value
* of the @{@link EnableAspectJAutoProxy#proxyTargetClass()} attribute on the importing
* {@code @Configuration} class.
*/
@Override
public void registerBeanDefinitions(
AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(registry);
//注册一个AOP代理实现的Bean
AnnotationAttributes enableAspectJAutoProxy =
AnnotationConfigUtils.attributesFor(importingClassMetadata, EnableAspectJAutoProxy.class);
if (enableAspectJAutoProxy.getBoolean("proxyTargetClass")) {
AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry);
}
if (enableAspectJAutoProxy.getBoolean("exposeProxy")) {
AopConfigUtils.forceAutoProxyCreatorToExposeProxy(registry);
}
}
}
spectJAutoProxyRegistrar 根据注解@EnableAspectJAutoProxy上的属性来注册相应的Bean,proxyTargetClass = true 指定使用CGLIB来作为AOP的代理实现方式。若不设置此属性,则默认使用jdk proxy作为代理实现方式。他们两者的区别是: jdk动态代理只支持接口的方法代理,不支持类自身的方法,它动态地生成一个代理类,继承自Proxy,实现被代理的接口,内部持有被代理对象;而CGLIB则支持所有的方法代理,它动态生成一个代理类,继承自被代理类,所以叫TargetClass。
AopConfigUtils
public abstract class AopConfigUtils {
//有三种AOP的配置形式,有基于Spring内置的,有基于用户配置文件的,有基于注解的,基于注解的优先级最高,
//一般配置一种就可以,如果配置多种则优先级高的会覆盖优先级低的
static {
APC_PRIORITY_LIST.add(InfrastructureAdvisorAutoProxyCreator.class);
APC_PRIORITY_LIST.add(AspectJAwareAdvisorAutoProxyCreator.class);
APC_PRIORITY_LIST.add(AnnotationAwareAspectJAutoProxyCreator.class);
}
public static BeanDefinition registerAspectJAnnotationAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry) {
return registerAspectJAnnotationAutoProxyCreatorIfNecessary(registry, null);
}
//注册AnnotationAwareAspectJAutoProxyCreator到容器中,此类负责基于注解的AOP动态代理实现
public static BeanDefinition registerAspectJAnnotationAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) {
return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);
}
//设置支持类的代理,设置此属性到时候会使用CGLIB来生成动态代理类
public static void forceAutoProxyCreatorToUseClassProxying(BeanDefinitionRegistry registry) {
if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
BeanDefinition definition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
definition.getPropertyValues().add("proxyTargetClass", Boolean.TRUE);
}
}
private static BeanDefinition registerOrEscalateApcAsRequired(Class<?> cls, BeanDefinitionRegistry registry, Object source) {
Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
if (!cls.getName().equals(apcDefinition.getBeanClassName())) {
int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());
int requiredPriority = findPriorityForClass(cls);
//通过比较代理类在集合中的序号来确定优先级,基于注解形式的优先级高
if (currentPriority < requiredPriority) {
apcDefinition.setBeanClassName(cls.getName());
}
}
return null;
}
RootBeanDefinition beanDefinition = new RootBeanDefinition(cls);
beanDefinition.setSource(source);
beanDefinition.getPropertyValues().add("order", Ordered.HIGHEST_PRECEDENCE);
beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
registry.registerBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME, beanDefinition);
return beanDefinition;
}
}
2.AspectJAutoProxyRegistrar被扫描过程(@ComponentScan工作过程)
ConfigurationClassParser#parse(AnnotationMetadata, java.lang.String)调用ConfigurationClassParser#processConfigurationClass调用
ConfigurationClassParser#doProcessConfigurationClass
里面通过@ComponentScans扫描到的 ,然后调用
ComponentScanAnnotationParser#parse调用
ClassPathBeanDefinitionScanner#doScan 把扫描到的所有的Bean 变成BeanDefinition对象
在ClassPathBeanDefinitionScanner#doScan 方法里面除了给所有的Bean变成
BeanDefinition之外,还会检查BeanDefinition是否有@Import或者@ComponentScan或者@ImportResources 等等,它又要去递归检测是否符合条件的 Bean变成BeanDefinition对象.
(二)XML方式设置方式(不要研究了)
AspectJAwareAdvisorAutoProxyCreator 是xml方式的入口类.xml配置
基于xml配置解析的方式,扫描Spring.xml 的
http\://www.springframework.org/schema/aop=org.springframework.aop.config.AopNamespaceHandler
点击去看到这个类里面有个init方法,找到下面代码块儿:
registerBeanDefinitionParser(“aspectj-autoproxy”, new AspectJAutoProxyBeanDefinitionParser());
就能看到解析
org.springframework.aop.config.AspectJAutoProxyBeanDefinitionParser
找到AspectJAutoProxyBeanDefinitionParser 类的
AspectJAutoProxyBeanDefinitionParser#parse调用
AopNamespaceUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary,这个方法就是xml的AOP入口类的注册的地方了
内部逻辑是:
———————————————————
registerAspectJAnnotationAutoProxyCreatorIfNecessary调用了
AopConfigUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary(org.springframework.beans.factory.support.BeanDefinitionRegistry, java.lang.Object) 调用了
AopConfigUtils#registerOrEscalateApcAsRequired方法注册了
AnnotationAwareAspectJAutoProxyCreator,把AnnotationAwareAspectJAutoProxyCreator这个类变成了BeanDefinition对象
———————————————————-