Spring AOP源码学习:AOP 注解的解析

正文

==

当使用 <aop:aspectj-autoproxy /> 注解开启 AOP 功能时。

Spring会从“META-INF/spring.handlers” 配置文件中拿到该注解对应的 NamespaceHandlerSupport:AopNamespaceHandler

在 AopNamespaceHandler 的 init 方法会给该注解注册对应的解析器,aspectj-autoproxy 对应的解析器是:AspectJAutoProxyBeanDefinitionParser。

@Override

public void init() {

// In 2.0 XSD as well as in 2.1 XSD.

registerBeanDefinitionParser(“config”, new ConfigBeanDefinitionParser());

registerBeanDefinitionParser(“aspectj-autoproxy”, new AspectJAutoProxyBeanDefinitionParser());

registerBeanDefinitionDecorator(“scoped-proxy”, new ScopedProxyBeanDefinitionDecorator());

// Only in 2.0 XSD: moved to context namespace as of 2.1

registerBeanDefinitionParser(“spring-configured”, new SpringConfiguredBeanDefinitionParser());

}

我们知道,当解析到 <aop:aspectj-autoproxy /> 注解时,会调用 AspectJAutoProxyBeanDefinitionParser 的 parse方法。

关于自定义注解的解析内容之前 IoC 的文章介绍过了,如果不了解的可以参考:Spring IoC:parseCustomElement 详解

AspectJAutoProxyBeanDefinitionParser#parse

==========================================

@Override

public BeanDefinition parse(Element element, ParserContext parserContext) {

// 1.注册AspectJAnnotationAutoProxyCreator

AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);

// 2.对于注解中子节点的处理

extendBeanDefinition(element, parserContext);

return null;

}

1.注册 AspectJAnnotationAutoProxyCreator,见代码块1。

代码块1:AopNamespaceUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary


public static void registerAspectJAnnotationAutoProxyCreatorIfNecessary(

ParserContext parserContext, Element sourceElement) {

// 1.注册AspectJAnnotationAutoProxyCreator

BeanDefinition beanDefinition = AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(

parserContext.getRegistry(), parserContext.extractSource(sourceElement));

// 2.对于proxy-target-class以及expose-proxy属性的处理

useClassProxyingIfNecessary(parserContext.getRegistry(), sourceElement);

// 3.注册组件并通知,便于监听器做进一步处理

registerComponentIfNecessary(beanDefinition, parserContext);

}

1.注册 AspectJAnnotationAutoProxyCreator,见代码块2。

2.对于 proxy-target-class 以及 expose-proxy 属性的处理,见代码块3。

代码块2:AopConfigUtils#registerAspectJAnnotationAutoProxyCreatorIfNecessary


public static BeanDefinition registerAspectJAnnotationAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) {

return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);

}

private static BeanDefinition registerOrEscalateApcAsRequired(Class<?> cls, BeanDefinitionRegistry registry, Object source) {

Assert.notNull(registry, “BeanDefinitionRegistry must not be null”);

// 1.如果注册表中已经存在beanName=org.springframework.aop.config.internalAutoProxyCreator的bean,则按优先级进行选择。

// beanName=org.springframework.aop.config.internalAutoProxyCreator,可能存在的beanClass有三种,按优先级排序如下:

// InfrastructureAdvisorAutoProxyCreator、AspectJAwareAdvisorAutoProxyCreator、AnnotationAwareAspectJAutoProxyCreator

if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {

// 拿到已经存在的bean定义

BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);

// 如果已经存在的bean的className与当前要注册的bean的className不相同,则按优先级进行选择

if (!cls.getName().equals(apcDefinition.getBeanClassName())) {

// 拿到已经存在的bean的优先级

int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());

// 拿到当前要注册的bean的优先级

int requiredPriority = findPriorityForClass(cls);

if (currentPriority < requiredPriority) {

// 如果当前要注册的bean的优先级大于已经存在的bean的优先级,则将bean的className替换为当前要注册的bean的className,

apcDefinition.setBeanClassName(cls.getName());

}

// 如果小于,则不做处理

}

// 如果已经存在的bean的className与当前要注册的bean的className相同,则无需进行任何处理

return null;

}

// 2.如果注册表中还不存在,则新建一个Bean定义,并添加到注册表中

RootBeanDefinition beanDefinition = new RootBeanDefinition(cls);

beanDefinition.setSource(source);

// 设置了order为最高优先级

beanDefinition.getPropertyValues().add(“order”, Ordered.HIGHEST_PRECEDENCE);

beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);

// 注册BeanDefinition,beanName为org.springframework.aop.config.internalAutoProxyCreator

registry.registerBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME, beanDefinition);

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
va开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!**

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值