spring源码学习_怎么学习spring框架源码-程序员宅基地

技术标签: spring  学习  java  

1.xmlBeanFactory对defaultListableBeanFactory类进行扩展,主要用于从XML文档中获取BeanDefinition,对于注册及获取bean都是使用从父类DefaultListableBeanFactory继承的方法去实现。
xmlBeanFactory 主要是使用reader属性对资源文件进行读取和注册。
2.循环依赖。创建bean,核心方法
getBean → doGetBean → createBean → doCreateBean → createBeanInstance → populateBean
3.BeanDefinitionReader获取资源文件,封装成BeanDefinition,通过BeanDefinitionRegistry将其注册起来,保存到BeanDefinitionMap中。BeanFactory根据bean定义信息,通过反射实例化bean。
4.FactoryBean,灵活创建bean。调用getObject方法,才会生成bean,但没有严格的bean生命周期流程
5.spring中有两种bean:容器bean(比如BFPP)、自定义bean
6.FactoryBean和BeanFactory

  • FactoryBean本身也是一个bean对象,是一个小工厂,能够生产另外的Bean。“&”+ beanName就是FactoryBean对象。
  • BeanFactory是spring的根接口,是大工厂,生成各种各样的bean。普通bean对象没有"&"
    在这里插入图片描述

7.PostProcessor

  • 后置处理器、增强器。有了它,我们可以在各个过程中,合理应用它,对bean定义信息进行扩展、修改
  • 针对beanFactory的后置处理器
    • BeanFactoryPostProcessor
    • BeanDefinitionRegistryPostProcessor 继承 BeanFactoryPostProcessor
  • 针对bean的后置处理器
    • BeanPostProcessor
    • InstantiationAwareBeanPostProcessor 继承 BeanPostProcessor

8.循环依赖-三级缓存

  • 只用二级缓存会将AOP中创建代理对象的时机提前,设计之初是让bean在生命周期的最后一步完成代理而不是在实例化后就马上完成
  • 循环依赖发生时提前代理,没有循环依赖代理方式不变,依然是初始化后代理
  • 如果只有1个缓存,能解决循环依赖吗?
  • 如果只有2个缓存,能解决循环依赖吗?
    • 可以,但是有aop的话,会违反设计理念,提前创建代理对象
  • 为什么使用3级缓存之后,就可以解决带aop的循环引用
    • aop是在实例化后、属性赋值(被调用)时,判断是否需要生成代理对象的。但是我们没有办法知道对象什么时候要被调用,所以用第3级缓存,保存lambda表达式,当对象被调用时,执行lambda表达式,判断是否返回代理对象
  • 为什么spring不能解决构造方法的循环依赖
    • 在Bean调用构造器实例化之前,一二三级缓存并没有Bean的任何相关信息,在实例化之后才放入三级缓存中,因此当getBean的时候缓存并没有命中,这样就抛出了循环依赖的异常了。
  • 为什么多例Bean不能解决循环依赖?
    • 我们的bean是单例的,而且是字段注入(setter注入)的,单例意味着只需要创建一次对象,后面就可以从缓存中取出来,字段注入,意味着我们无需调用构造方法进行注入。
    • 如果是原型bean,那么就意味着每次都要去创建对象,无法利用缓存;
    • 如果是构造方法注入,那么就意味着需要调用构造方法注入,也无法利用缓存。
  • 三级缓存介绍
    • 一级缓存的作用:存放可用的成品bean;
    • 二级缓存的作用:为了将成熟Bean和纯净Bean分离(未注入属性),避免多线程下读取到不完整的Bean;存放半成品bean,半成品bean即已经调用完构造但是还没有注入属性和初始化;
    • 三级缓存的作用:用来生产半成品的bean,与getbean方法解耦,能解决aop增强下的循环依赖;存放函数接口/钩子函数,函数接口实现创建动态代理调用BeanPostProcessor,即其要加强的aop处理(为了避免重复创建,调用会返回动态代理对象或者原实例,再存储在二级缓存);

真正的解决循环依赖是靠二级缓存,不用三级缓存也可以解决循环依赖,但这样就造成了在实例化后就立马完成代理,违背了最后一步完成代理的原则;
在创建bean的时候,在哪里通过什么方式创建了动态代理:通过BeanPostProcessor创建动态代理,在初始化之后或在出现循环依赖时实例化之后(实例化 -> 属性注入 -> 初始化)
发生循环依赖会用到二级缓存,普通依赖过程只用到一三级缓存

9.配置类加@Configuration 和不加的区别
@Bean中被依赖的bean不会被重复加载
@Configuration为Full配置类,经过了enhance增强处理,所有@Bean方法会被BeanMethodInterceptor拦截,根据方法名获取单例,而不是重复加载

10.Spring是怎样避免读取到不完整的Bean

防止多线程下Spring读取到不完整Bean加了两把锁
一把锁放在getSingleton()方法三级缓存,第二个线程阻塞直到第一个线程把二三级缓存删除完;
一把锁放在getSingleton(,)方法,先从单例池再拿一遍单例对象(double check防重复创建单例bean)
怎么样可以在所有Bean创建完后做扩展代码?
ContextRefreshedEvent/SmartInitializingSingleton

11.接口

  • BeanFactory

  • Aware

  • BeanDefition

  • BeanDefitionRegistry

    • 注册bean定义信息,对bean定义信息进行增删改查
  • BeanDefitionReader:bean定义信息读取器,解析xml、注解,转化为BeanDefinition,添加到BeanDefinitionMap中

  • BeanFactoryProcessor:bean工厂后置处理器,可以修改bean定义信息

  • BeanPostProcessor: bean后置处理器

  • Environment

    • StandardEnviroment :当完成整个程序的加载之后,会运行很多系统的一些属性值,添加到当前对象。后续程序运行的时候,直接从当前对象获取,而不需要每次都去读取
      • System.getEnv()
      • System.getProperties()
  • FactoryBean

    BeanFactory和factoryBean的区别都是用来创建对象的。当使用BeanFactory时,必须要遵循完整的创建过程。这个过程由spring管理。factoryBean只需要调用getObject就可以返回具体对象,整个对象创建过程由用户自己控制,更加灵活

    • isSingleton
    • getObject :用来返回具体对象。可以new,也可以反射。核心的点在于进行容器获取对象时,每次创建对象之前,都会提前做当前对象的一个判断
    • getObjectType
    • getBean(“&” + beanName) 就是获取工厂
    • 意义在于,我需要的时候,通过当前工厂里的getObject方法创建,而不需要遵循bean的生命周期

12.spring执行流程

  • ClassPathXmlApplicationContext构造方法
    • 调用父类构造方法,进行相关对象创建,初始化操作,准备好一些需要的集合、对象
    • 设置配置文件路径
    • 调用AbstractApplicationContext#refresh 13个方法
      • prepareRefresh:准备刷新工作
        • 设置开启时间
        • 设置开闭状态
        • initPropertySource 初始化属性资源
        • getEnvironment 获取环境对象
          • 父类构造方法中调用了customizePropertySource——定制化属性资源
        • 创建监听器、需要发布的事件集合,方便做扩展属性
      • obtainFreshBeanFacotry获得一个刷新的bean容器,创建当前工厂对象 DefualtListableBeanFactory,同时加载配置文件到属性值到当前工厂中,最重要的就是beanDefinition
        • refreshBeanFactory:如果有bean工厂,关闭,销毁,产生新的工厂
          • createBeanFactory
            • new DefualtListableBeanFactory
          • customizeBeanFacotry:是否允许被覆盖、是否允许被重写
          • loadBeanDefinitions:因为配置文件是多个的,这个方法有N多个重载的方法。把当前这些配置文件解析,封装成Beandefinition
        • getBeanFactory
      • prepareBeanFactory:给当前bean工厂设置属性值,完成bean工厂的初始化工作
        • 加载beanFactoryPostProcessor
        • 忽略aware接口实现,后面是在invokeAwareMethod中实现的
      • postProcessorFactory:后置处理相关操作,方法是空的(模板方法)
      • invokeBeanFactoryPostProcessor,实例化并且执行所有已经注册的BFPP
      • 准备实例化、初始化所需要的一些类、对象。创建bean
  • 几个重要的实现类
    • HierarchicalBeanFactory 获取父类工厂。在doGetBean方法中需要获取父类工厂
    • ListableBeanFactory,枚举所有bean实例
    • ConfigurableBeanFactory
    • DefaultListableBeanFactory
  • RootBeanDefinition
  • GenericBeanDefinition
  • getMergedBeanDefinition方法
    在这里插入图片描述
    源码图(调用关系)
    在这里插入图片描述

思维导图链接

spring执行流程总结

1.入口refresh();
2.实例化、初始化bean工厂,加载配置信息、资源、扫描包路径、注解,生成BeanDefinition对象,注册到BeanDefinitionMap中
3.注册BeanPostProcessor,执行BeanFactoryPostProcessor;
4.实例化bean(bean的生命周期)

  • 实例化bean对象
  • 属性注入
  • 初始化
    • 回调aware接口
    • 执行BeanPostBeforeProcessor
    • 如果实现了 InitializingBean接口,执行afterPropertiesSet方法
    • 如果有initMethod方法,就执行
    • 执行BeanPostAfterProcessor
  • 使用bean
  • 销毁bean
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/sinat_35803474/article/details/130811512

智能推荐

鸿蒙(HarmonyOS)性能优化实战-Trace使用教程

OpenHarmony的DFX子系统提供了为应用框架以及系统底座核心模块的性能打点能力,每一处打点即是一个Trace,其上附带了记录执行时间、运行时格式化数据、进程或线程信息等。开发者可以使用SmartPerf-Host调试工具对Trace进行解析,在其绘制的泳道图中,对应用运行过程中的性能热点进行分析,得出优化方案。本文旨在介绍OpenHarmony中常用的Trace,解释它们的含义和用途,并阐述如何通过这些Trace来识别潜在的性能问题。

服务器基础知识(2)

服务器基础知识

Python 实现京东自动登录领京豆_京东自动领京豆脚本-程序员宅基地

文章浏览阅读2.2w次,点赞4次,收藏54次。今天带大家进行模拟京东登录,并进行签到获取京豆,1000 个京豆 = 10 元,是不是一个发现了一个「发家致富」的好路子?废话不多说,下面开始正题。整体流程如下:京东自动签到流程1 模拟登录首先我们需要的就是模拟京东登录,只有登录了才能进行签到领京豆等操作。模拟登录其实就是通过 HTTP 的 POST 请求讲用户的登录信息发送给服务器进行认证的过程。1.1 登录数据分析登录过程表面上看着挺简单,我..._京东自动领京豆脚本

Google App Crash 参考解决方案,2024年最新拼多多面试java-程序员宅基地

文章浏览阅读565次,点赞17次,收藏20次。}--------- beginning of crash01-01 12:00:00.918 1583 1583 E AndroidRuntime: FATAL EXCEPTION: main01-01 12:00:00.918 1583 1583 E AndroidRuntime: Process: com.google.android.setupwizard, PID: 158301-01 12:00:00.918 1583 1583 E AndroidRuntime: java.lan

第一章:认识Java语言-程序员宅基地

文章浏览阅读916次,点赞16次,收藏19次。Java是一种简单、面向对象、分布式、稳健性、安全性、平台独立与可移植性、多线程、动态性的计算机编程语言。除了java还有很多编程语言:C语言、C++、C#、python等。不同的计算机编程语言语法不同;应用场景不同;Java是一种后端开发编程语言。开发流程:应用程序-》1、市场调研:用户的需求2、需求文档3、需求评审4、UI设计(提高用户体验)5、前端开发(利用前端技术实现网页,网页以浏览器直接打开\部署到浏览器,静态页面:数据不变)

Flutter浪潮下的音视频研发探索(1),android插件化-程序员宅基地

文章浏览阅读848次,点赞13次,收藏17次。UI这块知识是现今使用者最多的。当年火爆一时的Android入门培训,学会这小块知识就能随便找到不错的工作了。不过很显然现在远远不够了,拒绝无休止的CV,亲自去项目实战,读源码,研究原理吧!

随便推点

Zxing二维码生成并在dialog中显示_android zxing集成到 dialog-程序员宅基地

文章浏览阅读182次。文章目录二维码工具类自定义dialog,加载自定义布局使用二维码工具类import java.awt.image.BufferedImage;import java.io.File;import java.util.HashMap;import java.util.Hashtable;import java.util.Map;import javax.imageio.ImageIO;import android.content.Context;import android.graphics_android zxing集成到 dialog

云计算时代:SFP、SFP+、SFP28、QSFP+和QSFP28光纤模块详解

不同类型的光纤模块具有不同的数据传输速率、体积、功耗等特点,适用于不同的网络需求和环境。在众多光纤模块中,SFP、SFP+、SFP28、QSFP+和QSFP28是最常见的几种类型。SFP(Small Form Factor Pluggable)模块是一种小型可插拔的光纤模块,具有高速、高效率的特点。SFP+(Small Form Factor Pluggable Plus)模块是基于SFP模块的升级版,支持更高的数据传输速率,最高可达10Gbps。小巧灵活:SFP模块体积小巧,方便设备部署和更换。

VIL-SLAM论文翻译:Stereo Visual Inertial LiDAR Simultaneous Localization and Mapping-程序员宅基地

文章浏览阅读1.4k次,点赞2次,收藏18次。文章目录写在前面摘要1.引言2.相关工作3.系统概述4.视觉前端5.双目视觉惯性里程计A. IMU预积分因子B. 非结构化视觉因子C. 优化和边缘化6. 激光建图A. LiDAR扫描去畸变B. 帧到地图配准7. LiDAR增强的闭环A. 回环检测B. 回环约束C. 全局位姿图优化D. 重定位8. 实验结果A.平台和软件B.测试和结果C. EuRoCMAV数据集测试9.结论写在前面写作参考: robot L开源代码: 开源代码链接论文原文: 原文链接摘要SLAM是移动和空中机器人的一项基本任务_vil-slam

linux shell find命令 查找多种文件后缀_shell 同时查找两种结尾的配置文件-程序员宅基地

文章浏览阅读2.4w次,点赞14次,收藏46次。find命令最常用的是查找某个文件,如:find ./ -name "abc.txt"则会在当前目录及子目录下查找abc.txt文件更常用的是查找某一类型的文件,如:find ./ -name "*.txt"则会在当前目录及子目录下查找所有txt文件,但是如果要查找多种文件类型呢?比如某文件夹下面所有.c文件和.h文件,可以这样做:find ./ -name "*.[..._shell 同时查找两种结尾的配置文件

Rocky Linux安装部署Elasticsearch(ELK日志服务器)_rockylinux elk(1)-程序员宅基地

文章浏览阅读268次。(img-QrJpO645-1712860091908)]8、设置elasticsearch.service开机自动启动。10、查看elasticsearch.service运行状态。9、启动elasticsearch.service服务。7、使用yum安装elasticsearch。3、点击Downloads链接。5、查看yum安装方式。遇到一点意外,研究下先。4、点击左下角yum。

String中的intern()方法_string 的intern方法-程序员宅基地

文章浏览阅读349次。String中的intern方法的实现原理(jdk8中):以下面代码为例public class InternTest{ String s1 = new String("a") + new String("b"); s1.intern(); String s2 = "ab"; System.out.println(s1 == s2);//true }(1)String s1 = new String(“a”) + new String(“b”);①创建StringBuilder对_string 的intern方法