本文参考:https://www.bilibili.com/video/BV15b4y117RJ?p=176
- 关于代理的相关概念参考文章:java动态代理 ,详细介绍了什么是静态代理以及动态代理的原理。
- advice本质是实现advisor
- advice可以有多个通知,advisor只有一个
- methodintercepter属于环绕通知
- 注解形式的切面advisor无论@before还是@after都会转换成环绕通知并特殊处理
- 注解实现切面编程需要@aspect创建切面类,然后添加advisor(使用@around@before等等加上括号内的aspectj表达式去修修饰一个方法)也就是执行方法。
- annotationAwareAspectJAutoProxyCreator可以分为annotationaware(扫描注解)AspectJ(匹配aspectj表达式,绑定相对应的target类),autoproxyCreator(实现对应target类的代理创建)。总结:扫描注解当中的aspectj表达式找到对应target并为其创建代理对象。
- 二级缓存机制a在初始化的时候添加半成品到二级缓存当中(放入时间是a new的时候),singletonFactories(二级缓存)。a在使用b的时候先去查看二级缓存,二级缓存没有就调用b的getbean方法,b在new的时候放入到二级缓存当中一个半成品b,之后bseta的时候去二级缓存当中查找a,有a就接着走完自己的初始化了,最后返回完成的b给a再接着a的初始化
- 二级缓存在代理模式下存在漏洞:b所使用的a是非代理对象。
- 构造方法循环依赖用假的b推迟b创建,用@lazy修饰b找到代理对象,用工厂创建b的引用,provider类似与工厂,scope比较low指定生存范围。
循环依赖铺垫:代理对象
对下面的target1类进行功能增强也就是aop切面编程,实现代理
可以看到子类(target1的子类)代理对象调用目标方法时,执行的是子类代理对象的通知的方法invocation.proceed().在调用这个方法之前和之后实现了我们本身想要附加的代码业务也就是输出before和after。这就是代码功能增强。这也体现了功能不耦合的优点。
结合advisor使用切点
切点pointcut 可以实现对调用方法的过滤
如图只会执行foo的功能增强而不会增强bar方法
多个切面Advisor
输出结果
注解aop
下图中around是环绕通知,注解括号内就是切点。
首先@aspect创建切面类,然后添加advisor(使用@around@before等等)也就是执行方法
其中annotationAwareAspectJAutoProxyCreator可以分为annotationaware(扫描注解)AspectJ(匹配aspectj表达式,绑定相对应的target类),autoproxyCreator(实现对应target类的代理创建)。总结:扫描注解当中的aspectj表达式找到对应target并为其创建代理对象。
自动代理后处理器还提供了wrapifnecessary()方法。只有当匹配存在的aj表达式才会为其创建代理。
排除循环可能,不能为切面创建代理。图中排除了aspect1。
循环依赖
一级缓存
一级缓存不能解决循环依赖问题
看下面的红线
二级缓存
a在初始化的时候添加半成品到二级缓存当中(放入时间是a new的时候),singletonFactories(二级缓存)。a在使用b的时候先去查看二级缓存,二级缓存没有就调用b的getbean方法,b在new的时候放入到二级缓存当中一个半成品b,之后bseta的时候去二级缓存当中查找a,有a就接着走完自己的初始化了,最后返回完成的b给a再接着a的初始化。如下图:
二级缓存在代理模式下存在漏洞:b所使用的a是非代理对象。
三级缓存(spring叫做二级缓存:earlySingletonObjects)
三级缓存模式有了类工厂对象,这个工厂对象fa负责对循环依赖进行判断,有循环依赖,提前创建代理对象pa返回并且放到我们的三级缓存earlySingletonObjects当中(后续a会检查是否已经有了pa对象,那么就不再创建pa了直接放入到一级缓存)。
构造方法以及多例模式下的循环依赖
构造方法循环依赖用假的b推迟b创建,用@lazy修饰b找到代理对象,用工厂创建b的引用,provider类似与工厂,scope比较low指定生存范围。
lazy创建过程
工厂创建过程
provider接口
scope