Spring之@EnableAspectJAutoProxy开启AOP功能原理

Posted 敲代码的小小酥

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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功能的原理。

以上是关于Spring之@EnableAspectJAutoProxy开启AOP功能原理的主要内容,如果未能解决你的问题,请参考以下文章

风暴之Spring事务

我们一起学习Spring之Spring简介

Spring系列之手写一个SpringMVC

一文彻底解密Spring 源码之Spring MVC

java之spring之helloword

Spring之基础