我们日常会经常使用getBean()方法从spring容器中获取我们需要的bean。那么,getBean()方法是怎么一步步完成从spring容器中获取bean的呢?

下面我们就通过源码分析一下。

一、主要代码分析

  基于:spring5.1

 入口:AbstractBeanFactory类的doGetBean():

 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()方法的主要流程。

详解Spring中getBean()方法相关推荐

  1. 详解Spring的GetBean方法

    Spring容器提供了五种获取Bean的方式,可以根据beanName获取Bean,也可以根据classType来获取Bean,所有根据beanName来获取Bean的方式,底层都会通过调用下面的do ...

  2. 用IDEA详解Spring中的IoC和DI(挺透彻的,点进来看看吧)

    用IDEA详解Spring中的IoC和DI 一.Spring IoC的基本概念 控制反转(IoC)是一个比较抽象的概念,它主要用来消减计算机程序的耦合问题,是Spring框架的核心. 依赖注入(DI) ...

  3. 218.94.78.76:20001/index.php,详解spring中使用Elasticsearch的实例教程

    本篇文章主要介绍了详解spring中使用Elasticsearch的代码实现,具有一定的参考价值,有兴趣的可以了解一下 在使用Elasticsearch之前,先给大家聊一点干货. 1. ES和solr ...

  4. python expandtabs_详解Python中expandtabs()方法的使用

    详解Python中expandtabs()方法的使用 expandtabs()方法返回制表符,即该字符串的一个副本. '\t'已经使用的空间,可选择使用给定的tabsize(默认8)扩展. 语法 以下 ...

  5. 详解XMLHTTP中setRequestHeader方法和参数

    详解XMLHTTP中setRequestHeader方法和参数 setRequestHeader("Content-type", "application/x-www-f ...

  6. 详解Spring中@Autowire,@Value 注解实现原理

    点击关注公众号,实用技术文章及时了解 来源:blog.csdn.net/u013202238/article/details/107879726 本文主要基于SpringBoot-2.3.3.RELE ...

  7. 详解Spring中的CharacterEncodingFilter--forceEncoding为true在java代码中设置失效--html设置编码无效...

    在项目中有很多让人头疼的问题,其中,编码问题位列其一,那么在Spring框架中是如何解决从页面传来的字符串的编码问题的呢?下面我们来看看Spring框架给我们提供过滤器CharacterEncodin ...

  8. 详解Spring中Bean的自动装配~

    目录 1. 环境搭建 2. byName.byType 3. 使用注解实现自动装配 @Autowired @Resource 小结 自动装配是Spring满足bean依赖的一种方式 Spring会在上 ...

  9. Spring中getBean方法

    Spring源码阅读 : package com.example.riskm.core.common;import org.apache.commons.lang3.StringUtils; impo ...

最新文章

  1. 赠书:深入理解MySQL主从原理
  2. 【转】gif文件格式详解
  3. Linux文件权限基本配置
  4. Python之路(第三十一篇) 网络编程:简单的tcp套接字通信、粘包现象
  5. php printf 0.2f,php printf()
  6. matlab怎么整合成一个模块,Matlab如何将m文件制定成模块
  7. CSS光标cursor
  8. c 指针打印变量_C程序打印不同类型的指针变量的大小。
  9. Leetcode: Divide Two Integers
  10. 在 520 这天,竟然有人把 Docker讲清楚了? | 原力计划
  11. java 线程安全问题_java线程安全问题原因及解决办法
  12. 华为P50标准版规格曝光:或搭载骁龙888 4G
  13. 10-9-用户登录控制
  14. Failed to execute goal net.alchim31.maven:scala-maven-plugin:3.2.2:testCompile问题解决
  15. IOS第11天(4:UIDatePicker时间选择,和键盘处理,加载xib文件,代理模式)
  16. SD卡无法格式化怎么办?解决方法免费分享
  17. 即时通讯源码对企业到底有多重要呢?
  18. screen.colorDepth(色彩深度) 和 screen.pixelDepth(像素深度)
  19. 【转】bugku never give up 详解
  20. android go官方下载,GoFIT下载

热门文章

  1. 如何使用diskpart命令工具来对硬盘或者U盘设定写保护
  2. java画乌龟(略丑)
  3. Python初学13——jieba库简介与使用
  4. 白领的一天:请客吃饭实用口语
  5. SVM C++ 实现
  6. php 获取毫秒函数,PHP实现获取毫秒时间戳的方法【使用microtime()函数】
  7. Linux学习——redis主从配置(带密码)
  8. Python发现即将流失的客户
  9. 《Python编程:从入门到实践》第九章:类
  10. 实时语音识别选型的初步调研