当前位置: 首页 >Java技术 > 阿里四面:你知道Spring AOP创建Proxy的过程吗?

阿里四面:你知道Spring AOP创建Proxy的过程吗?

全是干货的技术号:
本文已收录在github,欢迎 star/fork:
https://github.com/Wasabi1234/Java-Interview-Tutorial

Spring在程序运行期,就能帮助我们把切面中的代码织入Bean的方法内,让开发者能无感知地在容器对象方法前后随心添加相应处理逻辑,所以AOP其实就是个代理模式。
但凡是代理,由于代码不可直接阅读,也是初级程序员们 bug 的重灾区。

1 案例

某游戏系统,含负责点券充值的类CouponService,它含有一个充值方法deposit():
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
deposit()会使用微信支付充值。因此在这个方法中,加入pay()。

由于微信支付是第三方接口,需记录接口调用时间。
引入 @Around 增强 ,分别记录在pay()方法执行前后的时间,并计算pay()执行耗时。
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
Controller:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
访问接口,会发现这段计算时间的切面并没有执行到,输出日志如下:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
切面类明明定义了切面对应方法,但却没执行到。说明在类的内部,通过this调用的方法,不会被AOP增强。

2 源码解析-创建代理对象的过程

  • this对应的对象就是一个普通CouponService对象:
    阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
  • 而在Controller层中自动装配的CouponService对象:
    阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
    是个被Spring增强过的Bean,所以执行deposit()时,会执行记录接口调用时间的增强操作。而this对应的对象只是一个普通的对象,并无任何额外增强。

为什么this引用的对象是普通对象?

这要从Spring AOP增强对象的过程来看。

AOP的底层是动态代理,创建代理的方式:

  • JDK方式
    只能对实现了接口的类生成代理,不能针对普通类
  • CGLIB方式
    可以针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法,来实现代理对象。
    阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记

针对非Spring Boot程序,除了添加相关AOP依赖项外,还会使用 @EnableAspectJAutoProxy 开启AOP功能。
这个注解类引入AspectJAutoProxyRegistrar,它通过实现ImportBeanDefinitionRegistrar接口完成AOP相关Bean准备工作。

  • 先看下调用栈:
    阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记

2.1 创建代理对象的时机

创建一个Bean时。

创建主要由AnnotationAwareAspectJAutoProxyCreator,一种BeanPostProcessor完成。
所以它的执行是在完成原始Bean构建后的初始化Bean(initializeBean)过程。

AbstractAutoProxyCreator

postProcessAfterInitialization

阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
关键方法wrapIfNecessary:在需要使用AOP时,它会把创建的原始Bean对象wrap成代理对象,作为Bean返回。

wrapIfNecessary

阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记

2.2 createProxy

创建代理对象的关键:

protected Object createProxy(Class<?> beanClass, @Nullable String beanName,  @Nullable Object[] specificInterceptors, TargetSource targetSource) {  // ...  // 1. 创建一个代理工厂  ProxyFactory proxyFactory = new ProxyFactory();  if (!proxyFactory.isProxyTargetClass()) {if (shouldProxyTargetClass(beanClass, beanName)) {  proxyFactory.setProxyTargetClass(true);}else {  evaluateProxyInterfaces(beanClass, proxyFactory);}  }  Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);  // 2. 将通知器(advisors)、被代理对象等信息加入到代理工厂  proxyFactory.addAdvisors(advisors);  proxyFactory.setTargetSource(targetSource);  customizeProxyFactory(proxyFactory);// ...// 3. 通过代理工厂获取代理对象  retu proxyFactory.getProxy(getProxyClassLoader());}

经过这样一个过程,一个代理对象就被创建出来了。我们从Spring中获取到的对象都是这个代理对象,所以具有AOP功能。而之前直接使用this引用到的只是一个普通对象,自然也就没办法实现AOP的功能了。

3 修正

经过分析可知,只有引用的是被 动态代理 所创对象,才能被Spring增强,实现期望的AOP功能

那得怎么处理对象,才具备这样的条件?

被@Autowired注解

通过 @Autowired,在类的内部,自己引用自己:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记

直接从AopContext获取当前Proxy

AopContext,就是通过一个ThreadLocal来将Proxy和线程绑定起来,这样就可以随时拿出当前线程绑定的Proxy。

使用该方案有个前提,需要在 @EnableAspectJAutoProxy 加配置项 exposeProxy = true ,表示将代理对象放入到ThreadLocal,这才可以直接通过

AopContext.currentProxy()

获取到,否则报错:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
于是修改代码:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记
勿忘修改EnableAspectJAutoProxyexposeProxy属性:
阿里四面:你知道Spring AOP创建Proxy的过程吗? _ JavaClub全栈架构师技术笔记

作者:JavaEdge.
来源链接:https://blog.csdn.net/qq_33589510/article/details/120387044

版权声明:
1、JavaClub(https://www.javaclub.cn)以学习交流为目的,由作者投稿、网友推荐和小编整理收藏优秀的IT技术及相关内容,包括但不限于文字、图片、音频、视频、软件、程序等,其均来自互联网,本站不享有版权,版权归原作者所有。

2、本站提供的内容仅用于个人学习、研究或欣赏,以及其他非商业性或非盈利性用途,但同时应遵守著作权法及其他相关法律的规定,不得侵犯相关权利人及本网站的合法权利。
3、本网站内容原作者如不愿意在本网站刊登内容,请及时通知本站(javaclubcn@163.com),我们将第一时间核实后及时予以删除。





本文链接:https://www.javaclub.cn/java/113471.html

标签:SpringAOP
分享给朋友:

“阿里四面:你知道Spring AOP创建Proxy的过程吗?” 的相关文章

SpringBoot整合消息队列工具kafka 2022年05月15日 21:58:17
常用锁原理的介绍(上) 2022年05月16日 18:33:06
SpringCloud系列之版本选择 2022年05月16日 20:39:20
全面了解 Java 原子变量类 2022年05月17日 20:52:45
Java实现阶乘运算 2022年05月21日 11:37:18
Spring Boot 2.0 WebFlux 快速入门实践 2022年05月27日 21:27:03
SpringBoot整合 mybatisPlus 2022年05月27日 21:55:24