马士兵

源码方法论

  1. 不要忽略源码中的注释
  2. 先梳理脉络,再深入细节
  3. 大胆猜测、小心求证
  4. 见名知意
  5. hold on
  6. 对源码有兴趣的都是变态
  7. 为了钱!

Spring

IoC

Spring容器帮助管理对象,不需要程序员去new。

<!-- applicationContext.xml -->
<bean id=? class=?>
<property name=? value=?/>
<property name=? ref=?/>
<bean/> <bean id=? class=?>
<constructor-arg name=? value=?/>
<constructor-arg name=? ref=?/>
<bean/>
  • 加载配置文件
  • 解析属性值
  • 创建实例对象
  • 使用
  • 销毁

Spring容器中使用 concurrentHashMap数据结构存放 bean。k-v结构存储。

Spring容器设计

常规创建对象的方式:

  1. new
  2. 反射
  3. 设计模式

Spring中 bean的作用域默认是单例的,其他还有 request的,session的。

// 常规反射创建对象
Class clazz = Class.forName();
Class clazz = 类名.class;
Class clazz = 对象名.getClass();
Constructor con = clazz.getDeclareConstructor();
Object obj = con.newInstance();

1)针对不同的配置方式 xml, 注释, json等有一个抽象定义规范接口 BeanDefinitonReader 读取Spring中 BeanDefinition接口中bean的定义信息。

2)

ApplicationContext ac = new ClassPathXmlApplicationContext();
Object o = ac.getBean("xxx");

实例化:在堆中开辟一块空间,属性都是默认值

初始化:给属性赋值、填充属性、执行初始化方法

<bean id="" class="" init-mmethod="" destroy-method="" />

完整对象

DefaultListableBeanFactory

从 BeanDefinition到 BeanFactory有多个 BeanFactoryPostProcessor实现过程增强(修改 bean的定义信息)。如:

<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="username" value="${jdbc.username}" />
</bean>

其中,在 <property/>标签中,将 value值从 db.properties中提取出来并赋值给username的过程便是一个过程增强的过程。具体调用的是 PlaceholderConfigurerSupport,它是 BeanFactoryPostProcessor的子类。

可以自定义类实现 BeanFactoryPostProcessor接口,获取 BeanDefinition,并对其进行修改。

应用场景:针对在 xml配置文件中定义了诸多的类并且发现部分同名属性都定义错了,就可以在容器读取配置文件后,通过 BeanFactoryPostProcessor进行截断修改。

针对注解开发的 @ComponentScan同样也需要有相应的增强器,ConfigurationClassPostProcessor,用于解析被注解的类。

bean的生命周期

1)实例化

2)填充属性(populateBean)

3)执行诸多 aware接口需要实现对方法

aware接口存在的意义是:方便通过 spring中的 bean对象来获取对应容器中的相关属性。

4)BeanPostProcessor(before, init-method, after)

CglibAopProxy

JdkDynamicAopProxy

5)完整的对象

6)销毁流程


观察者模式

在 Spring生命周期的不同阶段做不同的处理工作。

  1. 监听器
  2. 监听事件
  3. 多播器

Debug

1)ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");

2)Refresh()

3)BeanFactoryPostProcessor

4)BeanPostProcessor

5)多播器

6)监听器

7)实例化非懒加载的对象

doGetBean();

createBean();

doCreateBean();

createBeanInstance();

instabtiateBean();

getInstantiationStrategy().instantiate();

getDeclaredConstruction();

BeanUtils.instantiateClass();

newInstance();

populateBean();

invokeAwareMethods();

ignoreDependencyInterface(EnvironmentAware.class);

循环依赖

A类中定义了 b属性

B类中定义了 a属性

A调用B,B调用A

三级缓存

DefaultSingletonBeanRegistry

set注入:可以解决

构造方法注入:不能解决

原因:实例化和初始化分开处理(提前暴露对象)

实例化流程

1)先实例化 A对象,只是完成了堆空间开辟,并没有设置属性值,只是生成了半成品

2)初始化 A对象,此时要给 b属性赋值,而 B对象是一个完全独立的对象,所以此时要去 Spring容器中查找 B对象

如果此时已经有了 B对象,就直接赋值

如果此时没有 B对象,就先完成 B对象的实例化

3)实例化 B对象 。。。

4)初始化 B对象,给 B对象中的 a属性赋值 。。。

如果有了 A对象,就赋值

如果没有,就实例化 A对象

解决策略

  • 使用半成品
  • 放到容器内的一个 map中取用

ObjectFactory是一个函数式接口,仅有一个方法,可以传入 lambda表达式或者匿名内部类,可以通过调用 getObject()方法来执行具体的逻辑

1)创建 a:

​ getBean -> doGetBean -> getSingleton -> createBean -> doCreateBean

​ getObject()实际上调用的是 createBean();

​ instanceWrapper = nreateBeanInstance(beanName, mbd, args);

​ 得到对象 A@1755

2)填充 b:

​ getEarlyBeanReference(); 判断一级缓存内是否已经包含目标实例

​ 存入三级缓存:k-a, v-() ->getEarlyBeanReference(beanName, mbd, bean);

​ populateBean()

​ applyPropertyValus();开始填充属性值,填充 b

​ getName(); b

​ getValue(); RuntimeBeanReference@1983

​ resolveValueIfNecessary();

​ 判断 value类型是否为 RuntimeBeanReference -> resolveReference()

3)创建 b:B@2152

4)填充 b:存入三级缓存:k-b, v-() ->getEarlyBeanReference(beanName, mbd, bean);

populateBean(),给 b对象内的 a属性赋值 RuntimeBeanReference@2375

5)给 b中的 a赋值:在三级缓存中获取,获取 a所对应的 lambda表达式

此时 getObject() -> getEarlyBeanReference() -> 获取 A@1755

存入 二级缓存,删除 三级缓存中的 a对象,此时 a仍旧是半成品

此时 b的实例化和初始化已经完成

6)继续初始化 a:在一级缓存中存入k-b, v-B@2152,清除二级缓存和三级缓存中的 b

resolveValueIfNecessory()

initializeBean(); 之后 a就成为了完全体

在一级缓存中存入 k-a, v-A@1755,清除二级缓存和三级缓存中的 a

7)之后再需要获取时,直接在一级缓存中获取即可

查找顺序:

  1. 一级缓存:存完全体
  2. 二级缓存:存半成品
  3. 三级缓存:存 lambda,来完成代理对象的覆盖过程

总结

三级缓存解决循环依赖问题的关键是什么?

为什么通过提前暴露对象能解决?

因为对象的 实例化 和 初始化 分离

在中间过程中给其他对象赋值的时候,并不是一个完整的对象,而是把半成品对象赋值给了其他对象。

如果只使用一级缓存能不能解决问题?

不能!在整个调用过程中,一、二、三级缓存中存放的是半成品和成品对象,如果只有一级缓存,则成品和半成品都会放到一级缓存中。则有可能获取到半成品对象,而它是无法正常使用的。因此需要把半成品和成品对象的存放空间分割开来。

只使用二级缓存行不行?为什么需要三级缓存?

三级缓存相比二级缓存,多了个 getEarlyBeanReference();

如果,我们能保证所有的 bean对象都不需要此方法,那么只用二级缓存也可以。

getEarlyBeanReference():获取早期暴露的对象引用,解决循环依赖的关键!

AbstractAutoProxy();

createProxy();

使用三级缓存的本质在于解决AOP代理的问题!!

在创建代理对象时,如果某个 bean需要代理对象,那么会不会创建普通的 bean对象?

肯定会!

当调用 getEarlyBeanReference()时,会先把 bean赋值给 exposedBean,如果在之后的一系列判断中为 false,则直接返回该对象。否则,进行处理再返回。此时的 exposedObject

为什么使用了三级缓存就可以解决代理的问题?

当一个对象需要被代理的时候,在整个创建流程中包含两个对象(代理前的普通对象 和 代理后的牛逼对象)

而 bean默认是单例的,那么在整个生命周期的处理中,一个 beanname也只能对应一个对象。

解决措施:保证我在使用的时候,判断是否需要进行代理的处理。

怎么确定什么时候使用 AOP代理?

无法确定,因此需要使用匿名内部类或者 lambda,在使用的时候直接对普通对象进行覆盖。

保证全局唯一!

最新文章

  1. Linux与Windows xp操作系统启动过程
  2. CSS Animation
  3. MySQL安装,启动
  4. Shell编程—定时任务
  5. CoffeeScript学习(3)—— 函数
  6. 双有序队列算法——处理哈夫曼K叉树的高效算法
  7. python 3.6 import pymysql错误
  8. ZOJ 3635 Cinema in Akiba(线段树)
  9. 利用poi向excle写入数据
  10. 【转】10个重要的Linux ps命令实战
  11. CentOS 7 搭建基于携程Apollo(阿波罗)配置中心单机模式
  12. composer install 时遇到 Composer\Downloader\TransportException ...
  13. 网络编程基础【day09】:实现简单地ssh(四)
  14. vim的基本用法
  15. 转移动APP测试实践
  16. How to install VCM 2 Ford IDS 109 software
  17. 学习JavaScript计划
  18. CentOS下搭建Hadoop
  19. bcm53344 gpio驱动分析
  20. JDK动态代理代码示例

热门文章

  1. 4G巴歇尔槽流量采集网关
  2. Odoo 如何下载指定版本源码 &amp;&amp; .cfg配置参数
  3. 《吐血整理》进阶系列教程-拿捏Fiddler抓包教程(12)-Fiddler设置IOS手机抓包,你知多少???
  4. ajax.readyState与ajax.status一览
  5. 基于EasyExcel实现的分页数据下载封装
  6. 使用Python的selenium库制作脚本,支持后台运行
  7. C#基础_XML文件读写
  8. C# 数组 深拷贝 和 数组传参
  9. KingbaseES R6 集群sys_monitor.sh change_password一键修改集群用户密码
  10. 接口测试神器Apifox,亲测好用!