前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一文调试spring AOP源码

一文调试spring AOP源码

作者头像
简单的程序员
发布2020-07-15 09:53:36
4440
发布2020-07-15 09:53:36
举报
文章被收录于专栏:奕仁专栏奕仁专栏

在processOn上看到了aop的时序图,于是就看了看,整理了并写出此篇

在springboot中,开启AOP只需要加入如下注解

于是,追踪源码:

代码语言:javascript
复制
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
//根据spring体系的特殊功能,即注解具有继承性,于是可看出AspectJAutoProxyRegistrar类
@Import({AspectJAutoProxyRegistrar.class})
public @interface EnableAspectJAutoProxy {
    boolean proxyTargetClass() default false;

    boolean exposeProxy() default false;
}

点开AspectJAutoProxyRegistrar类之后,会发现这里会注入bean的定义信息,那么定义那个bean呢? 下面一起一探揪净~~

代码语言:javascript
复制
class AspectJAutoProxyRegistrar implements ImportBeanDefinitionRegistrar {
    AspectJAutoProxyRegistrar() {
    }

    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);
            }
        } 
    }
}

点开看看 原来是注册了org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator类,于是乎,点开源码看看咯,你瞧我这看源码的脾气~

这里,这个类继承了很多很多的类,这里用下面一张图来展示它的类依赖关系

你瞧,我的mac截图清晰不?哈哈,扯个dan。。。 看到上面的类依赖关系之后,经常玩spring的同学一下就可以看出下一步要追踪哪个类…… 对对对,没错,就是这个类,org.springframework.beans.factory.config.SmartInstantiationAwareBeanPostProcessor,这个类是啥意思呢?你木讷不?我不卖关子了,这个类就是在类实例化前做的一些操作,也就是如果beanMap不包含当前bean,就执行这里面的方法,具体实现自己可以去写个demo……知道这个就好了,断点就在org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#postProcessAfterInitialization方法,也就是在初始化之后调用这个方法,看代码:

代码语言:javascript
复制
@Override
	public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
		if (bean != null) {
		//bean肯定不为null 
			Object cacheKey = getCacheKey(bean.getClass(), beanName);
			//看它的缓存有没有,代理集合跟当前bean是不是一个  不是的话包装一下,也就是创建代理那一步
			if (this.earlyProxyReferences.remove(cacheKey) != bean) {
				return wrapIfNecessary(bean, beanName, cacheKey);
			}
		}
		return bean;
	}

下面就是看是否需要用代理包装一下当前bean,从方法命名也看得出来

代码语言:javascript
复制
protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {

	//上面都是对前几步写入的缓存进行校验  忽略
		if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {
			return bean;
		}
		if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {
			return bean;
		}
		//这一步 ……  看目标类是不是实现了Pointcut/Advisor/AopInfrastructureBean/Advice接口,如果实现了,直接返回
		//看目标类和beanName命名是否有关联
		if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {
			this.advisedBeans.put(cacheKey, Boolean.FALSE);
			return bean;
		}

		// Create proxy if we have advice.
		//好戏来了,上面判断完了之后,这里就要开始获取当前目标类对应的各种通知(前置/环绕通知,切面等)方法了,
		//,也就是匹配那几个注解了@pointcut,@Around等等 如果不等于代理,也就是返回的数组里面有切面和通知方法,
		//即给创建代理
		Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
		if (specificInterceptors != DO_NOT_PROXY) {
			this.advisedBeans.put(cacheKey, Boolean.TRUE);
			//创建代理的核心方法哦!!! 
			Object proxy = createProxy(
					bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));
			this.proxyTypes.put(cacheKey, proxy.getClass());
			return proxy;
		}

		this.advisedBeans.put(cacheKey, Boolean.FALSE);
		return bean;
	}

眼看马上要修成正果了~~ 创建代理的方法我单独贴出来去分析:

代码语言:javascript
复制
protected Object createProxy(Class<?> beanClass, @Nullable String beanName,
			@Nullable Object[] specificInterceptors, TargetSource targetSource) {
        //如果是beanFactory  暴露目标类
		if (this.beanFactory instanceof ConfigurableListableBeanFactory) {
			AutoProxyUtils.exposeTargetClass((ConfigurableListableBeanFactory) this.beanFactory, beanName, beanClass);
		}

		ProxyFactory proxyFactory = new ProxyFactory();
		proxyFactory.copyFrom(this);

		if (!proxyFactory.isProxyTargetClass()) {
		//是否可暴露目标类,也就是一开始我们注解的那两个属性
		//EnableAspectJAutoProxy(proxyTargetClass = true,exposeProxy = true)
			if (shouldProxyTargetClass(beanClass, beanName)) {
				proxyFactory.setProxyTargetClass(true);
			}
			else {
				evaluateProxyInterfaces(beanClass, proxyFactory);
			}
		}
		//构造它的aop切面拦截器>>>高能:这段代码就是为什么aop可以拦截到当前方法,包括各种通知
		Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);
		proxyFactory.addAdvisors(advisors);
		proxyFactory.setTargetSource(targetSource);
		customizeProxyFactory(proxyFactory);

		proxyFactory.setFrozen(this.freezeProxy);
		if (advisorsPreFiltered()) {
			proxyFactory.setPreFiltered(true);
		}
	  //生成代理  
		return proxyFactory.getProxy(getProxyClassLoader());
	}

生成代理这块我没有去追踪源码,而我用一张时序图来对生成代理的过程进行查看~ 也就是下面这张图

Spring AOP 源码解析时序图.png

到了这里,aop的初始化的一部分原理就调试完了,那么他为什么要这么做呢?有什么好处呢?他可以用到哪些场景?

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2020-07-14,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
云顾问
云顾问(Tencent Cloud Smart Advisor)是一款提供可视化云架构IDE和多个ITOM领域垂直应用的云上治理平台,以“一个平台,多个应用”为产品理念,依托腾讯云海量运维专家经验,助您打造卓越架构,实现便捷、灵活的一站式云上治理。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档