当我们通过向Spring容器获取某个bean的时候,总是调用Spring中重载的各种getBean方法。那么,getBean中的流程是什么样的?

通过本文,你将对getBean方法的主流程有一个详细的认识。

入口当然是getBean方法:

 public Object getBean(String name) throws BeansException {return doGetBean(name, null, null, false);}protected <T> T doGetBean(final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly)  throws BeansException {//beanName转换,这里主要有两方面的考虑://1:如果beanName是以&开头的,则表明是要返回FactoryBean本身//而不是其生产出来的对象,而FactoryBean存储的时候,跟普通的bean一样//2:如果传入的是别名,则是需要转换为实际的beanName的           final String beanName = transformedBeanName(name);Object bean;// 检查bean是否已经存在了缓存中Object sharedInstance = getSingleton(beanName);//不为空则表明之前被创建过if (sharedInstance != null && args == null) {if (logger.isDebugEnabled()) {.............}//这里对于普通的bean,则会直接的返回,//如果是FactoryBean类型的则会创建对应的实例返回bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);}else {//如果是正在创建的Prototype类型的bean,无法处理该类型循环依赖的问题,则直接抛出异常信息      if (isPrototypeCurrentlyInCreation(beanName)) {throw new BeanCurrentlyInCreationException(beanName);}// 查看父类中是否有相关的bean的定义信息BeanFactory parentBeanFactory = getParentBeanFactory();if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {// 如果name是以&开头的  则返回&+beanName的格式String nameToLookup = originalBeanName(name);if (args != null) {//递归去父类中查找return (T) parentBeanFactory.getBean(nameToLookup, args);}else {//递归去父类中查找return parentBeanFactory.getBean(nameToLookup, requiredType);}}if (!typeCheckOnly) {markBeanAsCreated(beanName);}try {//将父类的定义的BeanDefinition与子类的BeanDefinition进行合并覆盖final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);checkMergedBeanDefinition(mbd, beanName, args);//判断bean是否有dependsOn 类的依赖//如果没有循环依赖,则先创建所dependsOn依赖的beanString[] dependsOn = mbd.getDependsOn();if (dependsOn != null) {for (String dep : dependsOn) {//这里主要是判断是否有以下这种类型的依赖://<bean id="beanA" class="BeanA" depends-on="beanB">//<bean id="beanB" class="BeanB" depends-on="beanA">//如果有,则直接抛出异常if (isDependent(beanName, dep)) {.......}registerDependentBean(dep, beanName);getBean(dep);}}// 处理完bean的依赖关系,则开始来创建beanif (mbd.isSingleton()) {//创建单例的bean,这里的createBean下文将详细讲解sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() {@Overridepublic Object getObject() throws BeansException {try {return createBean(beanName, mbd, args);}catch (BeansException ex) {destroySingleton(beanName);throw ex;}}});bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);}else if (mbd.isPrototype()) {//创建原型的beanObject prototypeInstance = null;try {beforePrototypeCreation(beanName);prototypeInstance = createBean(beanName, mbd, args);}finally {afterPrototypeCreation(beanName);}bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);}else {//创建其他生命周期的beanString scopeName = mbd.getScope();final Scope scope = this.scopes.get(scopeName);if (scope == null) {.........}try {Object scopedInstance = scope.get(beanName, new ObjectFactory<Object>() {@Overridepublic Object getObject() throws BeansException {beforePrototypeCreation(beanName);try {return createBean(beanName, mbd, args);}finally {afterPrototypeCreation(beanName);}}});bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);}catch (IllegalStateException ex) {........}}}catch (BeansException ex) {cleanupAfterBeanCreationFailure(beanName);throw ex;}}// bean的类型转换if (requiredType != null &&
bean != null && !requiredType.isAssignableFrom(bean.getClass())) {try {return getTypeConverter().convertIfNecessary(bean, requiredType);}catch (TypeMismatchException ex) {........}}return (T) bean;}

整个getBean大的流程如上所示,在以上源码中,我们将看下以下的几个的方法:

1:transformedBeanName方法:

 protected String transformedBeanName(String name) {//如果beanName是以&开头的  则截取掉开头的&return canonicalName(BeanFactoryUtils.transformedBeanName(name));}public String canonicalName(String name) {String canonicalName = name;// 循环的去获取别名,直到获取到真实的beanName//aliasA->aliasB->beanNameString resolvedName;do {resolvedName = this.aliasMap.get(canonicalName);if (resolvedName != null) {canonicalName = resolvedName;}}while (resolvedName != null);return canonicalName;}

2:getSingleton方法

 public Object getSingleton(String beanName) {return getSingleton(beanName, true);}//判断是否有缓存起来的beanprotected Object getSingleton(String beanName, boolean allowEarlyReference) {//singletonObjects保存的为实例化并赋值过的bean,可以直接使用Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {synchronized (this.singletonObjects) {//earlySingletonObjects主要是保存允许提前暴露出来的bean//主要是为了解决循环依赖的问题singletonObject = this.earlySingletonObjects.get(beanName);if (singletonObject == null && allowEarlyReference) {ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);if (singletonFactory != null) {singletonObject = singletonFactory.getObject();this.earlySingletonObjects.put(beanName, singletonObject);this.singletonFactories.remove(beanName);}}}}return (singletonObject != NULL_OBJECT ? singletonObject : null);}

3:getObjectForBeanInstance方法

 protected Object getObjectForBeanInstance(Object beanInstance, String name, String beanName, RootBeanDefinition mbd) {//如果name是以&开头的 但是不是FactoryBean,则直接抛出异常if (BeanFactoryUtils.isFactoryDereference(name) && !(beanInstance instanceof FactoryBean)) {throw new BeanIsNotAFactoryException(transformedBeanName(name), beanInstance.getClass());}// 则beanInstance可能是一个普通的bean,也可能是一个FactoryBean// 如果是一个普通的bean,则直接返回if (!(beanInstance instanceof FactoryBean) || BeanFactoryUtils.isFactoryDereference(name)) {return beanInstance;}//FactoryBean创建出bean实例返回Object object = null;if (mbd == null) {object = getCachedObjectForFactoryBean(beanName);}if (object == null) {FactoryBean<?> factory = (FactoryBean<?>) beanInstance;if (mbd == null && containsBeanDefinition(beanName)) {mbd = getMergedLocalBeanDefinition(beanName);}boolean synthetic = (mbd != null && mbd.isSynthetic());object = getObjectFromFactoryBean(factory, beanName, !synthetic);}return object;}

以上对getBean主流程以及其主要的几个方法做了分析,下面将对其getBean的主流程做一个分析:

通过流程图,将其主要的流程的轮廓给描述的出来。更加的直观。在下文中,我将对Spring中createBean做一个详细的描述

Spring源码分析之getBean主流程分析相关推荐

  1. Spring源码解析之-- 事务TransactionInterceptor 分析(开启事务)

    目录 一.介绍 二.TransactionInterceptor 分析 2. 流程 2.1 invoke 2.1.1 TransactionAspectSupport#invokeWithinTran ...

  2. spring源码阅读--aop实现原理分析

    aop实现原理简介 首先我们都知道aop的基本原理就是动态代理思想,在设计模式之代理模式中有介绍过这两种动态代理的使用与基本原理,再次不再叙述. 这里分析的是,在spring中是如何基于动态代理的思想 ...

  3. Spring源码讲解之 getBean 方法

    getBean 方法是用来进行获取 bean 实例对象的,是我们进行 spring 程序开发从 bean 工厂获取结果的.那这个方法的执行过程是什么样的,下面我们就去揭开它的面纱. 样例代码 Appl ...

  4. Spring源码之BeanFactory.getBean()

    本文是针对Srping的BeanFactory.getBean来进行源码解析,如果您是第一次看请先看一下XMLBeanFactory解析:https://blog.csdn.net/qq_302571 ...

  5. Spring源码解析之getBean(beanName)

    获取Spring Bean大体步骤: 调用BeanDefinitionReader的loadBeanDefinitions()方法解析并加载bean配置信息,转换位BeanDefinition对象 讲 ...

  6. 首发,看了这份美团架构师的spring源码笔记后,才发现原来学习的思路都错了

    前言 Spring让我们可以更快,更轻松,更安全地进行Java编程.Spring对速度,简单性和生产率的关注使其成为世界上最受欢迎的Java框架. 像阿里巴巴,亚马逊,微软等在内的所有科技巨头对Spr ...

  7. 【Spring源码分析】Bean加载流程概览

    代码入口 之前写文章都会啰啰嗦嗦一大堆再开始,进入[Spring源码分析]这个板块就直接切入正题了. 很多朋友可能想看Spring源码,但是不知道应当如何入手去看,这个可以理解:Java开发者通常从事 ...

  8. Spring源码分析-深入理解生命周期之BeanFactoryProcessor

    生命周期之BeanFactoryPostProcessor 先来看看bean的生命周期.对于熟悉spring 的朋友来说,bean的生命周期并不陌生.它可以在bean加载,bean初始化的过程中加入我 ...

  9. spring 源码分析(1)-xml文件解析

    我们在最开始接触spring的时候,看到不少书spring入门的例子如下 ApplicationContext atx = new ClassPathXmlApplicationContext(&qu ...

最新文章

  1. 介绍一个团队协同管理工具:Leangoo
  2. 华为对边缘计算的思考与理解
  3. 通过电机编码器AB相输出确定电机转向
  4. Leetcode - 347. Top K Frequent Elements(堆排序)
  5. 定位系列论文阅读-RoNIN(二)-Robust Neural Inertial Navigation in the Wild: Benchmark, Evaluations
  6. VMP分析之VM解码循环与基本架构(一)
  7. mysql pt-kill_percona-toolkit之pt-kill 杀掉mysql查询或连接的方法
  8. linux查看信息文件,lsof---Linux查看文件信息的强大工具
  9. python识别手写文字_如何快速使用Python神经网络识别手写字符?(文末福利)
  10. [转帖]win10 .Net Runtime Optimization Service占用大量CPU资源解决方法
  11. Python基础----NumPy
  12. sql 查询数据长度
  13. Laravel5.1/Homestead (0.2.7) 开发环境的部署和设置
  14. MATLAB程序界面介绍
  15. rust修改礼包时间_Rust Async: futures-timer源码解析
  16. SolidWorks用鼠标中键控制模型的旋转、缩放和平移
  17. 【一起入门NLP】中科院自然语言处理第13课-NLP基础任务④:序列生成问题(Seq2Seq)
  18. acwing 206 石头游戏 矩阵快速幂
  19. 分享几个实用的chatgpt网址
  20. 你知道八皇后嘛?就是那个八皇后

热门文章

  1. 数据结构 图9 关键活动
  2. 用java的输出姓名_用java程序输出自己的姓名
  3. i技术会 | 大数据应用与数据中台实践
  4. 零基础学习Linux必会的60个常用命令
  5. SOUI的一个动态创建控件的小例子
  6. Vuex的作用、使用、核心概念(State、Mutations、Getters、Actions、Modules)、文件抽离
  7. 会员模块(会员注册、会员登录、忘记密码、会员中心)
  8. Kali NetHunter在小米5上的环境安装搭建
  9. 通过AT命令使用4G模块通讯软件设计
  10. android短彩信数据库设计源码解析(二)