Spring之@EnableAspectJAutoProxy开启AOP功能原理

疑问

在使用AOP功能时,我们需要用@EnableAspectJAutoProxy手动开启AOP功能,否则不生效,为什么写了@EnableAspectJAutoProxy注解就可以使用AOP功能了呢,其中的原理是什么呢?

答案

在Spring中,无论是AOP功能还是其他需要手动开启的功能,实现思路大致如下:
开启功能的注解中,@Import了一个类,这个类都是BeanDefinitionPostprocessor类型的对象,在这个对象中,把开启相关功能的入口类,加入到了Spring容器中,所以就具有了该功能。而没有开启注解的话,相关功能的入口类就没有进入Spring容器,Spring容器也就没有相关功能的判断和处理。

原理解析

我们以AOP的@EnableAspectJAutoProxy为例,分析上述回答的原理。
在这里插入图片描述
首先,@EnableAspectJAutoProxy import进了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);

		AnnotationAttributes enableAspectJAutoProxy =
				AnnotationConfigUtils.attributesFor(importingClassMetadata, EnableAspectJAutoProxy.class);
		if (enableAspectJAutoProxy != null) {
			if (enableAspectJAutoProxy.getBoolean("proxyTargetClass")) {
				AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry);
			}
			if (enableAspectJAutoProxy.getBoolean("exposeProxy")) {
				AopConfigUtils.forceAutoProxyCreatorToExposeProxy(registry);
			}
		}
	}

}

可以看到这个类是一个BeanDefinitionRegistrar类,看其registerBeanDefinitions方法,看注册了什么beandefition进来。
在这里插入图片描述
看registerAspectJAnnotationAutoProxyCreatorIfNecessary方法,一顿点入方法后,到了如下方法:

private static BeanDefinition registerOrEscalateApcAsRequired(
			Class<?> cls, BeanDefinitionRegistry registry, @Nullable 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;
	}

这个方法重点看往register中注册了什么,即如下方法:
在这里插入图片描述
通过断点可以看到:
在这里插入图片描述
注册了AnnotationAwareAspectJAutoProxyCreator类,我们看这个类的结构图:
在这里插入图片描述
所以,这里注册了AOP的入口类,而AOP的入口类又是一个BeanpostProcessor对象,这样,就有了AOP的功能,并听过BeanPostProcessor对相关的bean进行了代理操作。
这就是开启AOP功能的原理。

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

敲代码的小小酥

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值