栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

Spring之@EnableAspectJAutoProxy开启AOP功能原理

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring之@EnableAspectJAutoProxy开启AOP功能原理

疑问

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

答案

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

原理解析

我们以AOP的@EnableAspectJAutoProxy为例,分析上述回答的原理。

首先,@EnableAspectJAutoProxy import进了AspectJAutoProxyRegistrar类,我们看这个类:

class AspectJAutoProxyRegistrar implements importBeanDefinitionRegistrar {

	
	@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功能的原理。

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/322178.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号