Spring3.1.0实现原理分析(八).获取bean对象

Stella981
• 阅读 665

     Spring获取bean对象的逻辑相对简单,原本不打算专门写篇博客来介绍了。不过想了想,既然上两篇博客分析了Spring创建对象的过程,有创建便有获取,所以还是分析下获取bean对象的过程吧。

一. 获取单例bean

1). 进入DefaultSingletonBeanRegistry的getSingleton方法,下面是代码片段,简单分析下。

    protected Object getSingleton(String beanName, boolean allowEarlyReference) 
    {
        //步骤a
        Object singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject == null) 
        {
            synchronized (this.singletonObjects) 
            {
                //步骤b
                singletonObject = this.earlySingletonObjects.get(beanName);
                if (singletonObject == null && allowEarlyReference) 
                {
                    ObjectFactory singletonFactory = this.singletonFactories.get(beanName);
                    if (singletonFactory != null) 
                    {
                        //步骤c
                        singletonObject = singletonFactory.getObject();
                        this.earlySingletonObjects.put(beanName, singletonObject);
                        this.singletonFactories.remove(beanName);
                    }
                }
            }
        }
        return (singletonObject != NULL_OBJECT ? singletonObject : null);
    }

    ** a.** 首先尝试从本类的成员变量Map<String, Object> singletonObjects中,根据bean's Id获取单例bean对象。(其实这个map就是Spring启动过程中创建的单例bean对象缓存)。

          a.1.通常可以从singletonObjects中获取到单例对象,如果得到bean对象实现了FactoryBean接口,并且引用的Id前面没加"&"符号,则调用FactoryBean接口的getObject()方法创建对象并返回。

          a.2.如果singletonObjects中不存在,比如启动时首次创建,则继续执行步骤b。

     b. 尝试从本类的另外一个成员变量Map<String, Object> earlySingletonObjects中根据bean's Id获取。(这个map中缓存的bean对象是“半成品”,实例化是成功了,但是卡在填充属性值这个步骤,原因是该bean依赖的bean尚为创建成功,为了实现单例bean基于setter的循环引用,所以Spring把这种半成品缓存起来,然后这个缓存中的bean对象其实是通过调用下述步骤C的ObjectFactory接口创建的。关循环依赖可以看下这篇博客关于循环引用的探讨)。 如果earlySingletonObjects中存在,则返回反之继续执行。

     c. 再尝试从本类成员变量Map<String, ObjectFactory> singletonFactories中根据bean's Id获取对象工厂。如果存在ObjectFactory,则调用其getObject()方法获取半成品的bean对象。在getObject()方法内部,如果当前bean工厂存在"聪明实例化敏感bean后处理器"(SmartInstantiationAwareBeanPostProcessor),会调用bean后处理器getEarlyBeanReference方法对bean执行后处理,如果不存在bean后处理器,则直接返回bean对象。

    d. 把步骤c获取的bean对象置入Map<String, Object> earlySingletonObjects中。

二. 获取属性bean

这个就很简单了,每次都重新创建bean对象,具体可以看前两篇博客,详细分析了创建bean对象的完整过程。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
从源码层面深度剖析Spring循环依赖
作者:郭艳红以下举例皆针对单例模式讨论图解参考1、Spring如何创建Bean?对于单例Bean来说,在Spring容器整个生命周期内,有且只有一个对象。Spring在创建Bean过程中,使用到了三级缓存,即DefaultSingletonBeanRegi
Stella981 Stella981
3年前
Spring3.1.0实现原理分析(九).AOP之创建代理对象的过程
    大家好,今天我会用一个例子来讲解Spring创建bean代理对象的过程,为大家揭开SpringAOP的神秘面纱。在看这篇博客前我强烈建议读者先看下这两篇博客《Spring3.1.0实现原理分析(六).实例化》(https://www.oschina.net/action/GoToLink?urlhttp%3A%2F%2Fhttp%2F
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Easter79 Easter79
3年前
Spring循环依赖问题的解决
循环依赖问题一个bean的创建分为如下步骤:!(https://static.oschina.net/uploads/img/202102/24030007_IrhH.png)当创建一个简单对象的时候,过程如下:先从单例池中获取bean,发现无a创建a的实例为a赋值把a放到单例池
Wesley13 Wesley13
3年前
Spring方法注入
在spring中注入方式有3中:1,构造函数注入2,set方法注入3,接口注入(方法注入)在spring中的bean默认范围都是单例,但是在特定的情况下,我们需要有如下的业务需要,单例bean1需要依赖非单例bean2,由于bean1始终是单例,所以如果不做出改变,每次获取的bean2也是同一个,容器就没办法给我们提供