Spring源码:Bean生命周期(三)

前言

在之前的文章中,我们已经对 bean 的准备工作进行了讲解,包括 bean 定义和 FactoryBean 判断等。在这个基础上,我们可以更加深入地理解 getBean 方法的实现逻辑,并在后续的学习中更好地掌握createBean 方法的实现细节。

getBean用法

讲解getBean方法之前,我们先来看看他有几种常见的用法:

// 创建一个Spring容器  
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);  
UserService bean1 = applicationContext.getBean(UserService.class);  
UserService bean2 = (UserService)applicationContext.getBean("userService");  
UserService bean3 = applicationContext.getBean("userService",UserService.class);  
UserService bean4 = (UserService) applicationContext.getBean("userService",new OrderService());  
bean1.test();  
bean2.test();  
bean3.test();  
bean4.test();

关于获取 bean 的方法,前两种方法应该比较常见,这里就不再赘述。第三种方法实际上是在获取 bean 的时候,会先判断是否符合指定的类型,如果符合,则进行类型转换并返回对应的 bean 实例。第四种方法则是在创建 bean 实例时,通过推断构造方法的方式来选择使用带有参数的构造方法进行实例化。

如果我们想要让第四种方法生效,可以考虑使用多例的形式,即通过设置 scope 属性为 prototype 来实现。这样,每次获取 bean 时,都会创建新的 bean 实例,从而可以触发使用带有参数的构造方法进行实例化,比如这样:

@Component  
@Scope("prototype")  
public class UserService {  
  
     public UserService(){  
      System.out.println(0);  
   }  
     public UserService(OrderService orderService){  
      System.out.println(1);  
   }  
   public void test(){  
      System.out.println(11);  
   }  
}

getBean大体流程

由于方法代码太多,我就不贴代码了,我这边只贴一些主要的伪代码,方便大家阅读,然后我在对每个流程细讲下:

	protected <T> T doGetBean(
			String name, @Nullable Class<T> requiredType, @Nullable Object[] args, boolean typeCheckOnly)
			throws BeansException {

		// name有可能是 &xxx 或者 xxx,如果name是&xxx,那么beanName就是xxx
		// name有可能传入进来的是别名,那么beanName就是id
		String beanName = transformedBeanName(name);
		Object beanInstance;

		// Eagerly check singleton cache for manually registered singletons.
		Object sharedInstance = getSingleton(beanName);
		if (sharedInstance != null && args == null) {
			
			// 如果sharedInstance是FactoryBean,那么就调用getObject()返回对象			
		}

		else {	
			//检查是否本beanfactory没有当前bean定义,查看有父容器,如果有,则调用父容器的getbean方法				  
			try {				 
				RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
				// 检查BeanDefinition是不是Abstract的
				checkMergedBeanDefinition(mbd, beanName, args);
				// Guarantee initialization of beans that the current bean depends on.				 
				//查看是否有dependsOn注解,如果存在循环依赖则报错
				// Create bean instance.
				if (mbd.isSingleton()) {
					//调用createBean方法
					//如果是FactoryBean则调用getObject
				}
				else if (mbd.isPrototype()) {
					//调用createBean方法,与单例只是前后逻辑不一样
					//如果是FactoryBean则调用getObject
				}
				else {
					Scope不同类型有不同实现
					//调用createBean方法,与单例只是前后逻辑不一样
					//如果是FactoryBean则调用getObject
				}
			}catch{
				......
			}
		}

		// 检查通过name所获得到的beanInstance的类型是否是requiredType
		return adaptBeanInstance(name, beanInstance, requiredType);
	}

单例缓存池

在 Spring 中,不管传入的 beanName 是多例的还是单例的,都会先从单例缓存池中获取。有些人可能会觉得这样做会浪费一些性能,但实际上 Spring 考虑到了大部分托管的 bean 都是单例的情况,因此忽略了这一点性能。实际上,这样的性能消耗并不大。可以将其类比于 Java 的双亲委派机制,都会先查看本加载器是否有缓存,如果没有再向父加载器去加载。

parentBeanFactory

在分析 bean 定义是如何创建的时,我们可以不考虑单例缓存池中获取对象的情况,而是逐步分析 bean 定义是如何创建的。在这个过程中,即使存在 parentBeanFactory,我们也可以跳过它,因为我们的启动容器并没有设置任何父容器。源码也很简单,如果本容器没有 bean 定义,就直接调用父容器的 getBean 相关方法:

BeanFactory parentBeanFactory = getParentBeanFactory();
            if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
                // Not found -> check parent.
                // &&&&xxx---->&xxx
                String nameToLookup = originalBeanName(name);
                if (parentBeanFactory instanceof AbstractBeanFactory) {
                    return ((AbstractBeanFactory) parentBeanFactory).doGetBean(
                            nameToLookup, requiredType, args, typeCheckOnly);
                }
                else if (args != null) {
                    // Delegation to parent with explicit args.
                    return (T) parentBeanFactory.getBean(nameToLookup, args);
                }
                else if (requiredType != null) {
                    // No args -> delegate to standard getBean method.
                    return parentBeanFactory.getBean(nameToLookup, requiredType);
                }
                else {
                    return (T) parentBeanFactory.getBean(nameToLookup);
                }
            }

dependsOn

在调用 getBean 方法之前,已经将合并的 bean 定义存入了容器中。因此,我们可以直接获取已经合并好的 bean 定义,并解析 bean 定义上的 dependsOn 注解。具体的源码逻辑如下:

RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);

                // 检查BeanDefinition是不是Abstract的
                checkMergedBeanDefinition(mbd, beanName, args);

                // Guarantee initialization of beans that the current bean depends on.
                String[] dependsOn = mbd.getDependsOn();
                if (dependsOn != null) {
                    // dependsOn表示当前beanName所依赖的,当前Bean创建之前dependsOn所依赖的Bean必须已经创建好了
                    for (String dep : dependsOn) {
                        // beanName是不是被dep依赖了,如果是则出现了循环依赖
                        if (isDependent(beanName, dep)) {
                            throw new BeanCreationException(mbd.getResourceDescription(), beanName,
                                    "Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
                        }
                        // dep被beanName依赖了,存入dependentBeanMap中,dep为key,beanName为value
                        registerDependentBean(dep, beanName);

                        // 创建所依赖的bean
                        try {
                            getBean(dep);
                        }
                        catch (NoSuchBeanDefinitionException ex) {
                            throw new BeanCreationException(mbd.getResourceDescription(), beanName,
                                    "'" + beanName + "' depends on missing bean '" + dep + "'", ex);
                        }
                    }
                }

这里的逻辑还是相对简单的。如果当前 bean 被 dependsOn 注解所依赖,那么会先去创建所依赖的 bean。但是这种方式是解决不了循环依赖的问题的。在实现上,只使用了两个 Map 进行判断:

// 某个Bean被哪些Bean依赖了
private final Map<String, Set<String>> dependentBeanMap = new ConcurrentHashMap<>(64);
// 某个Bean依赖了哪些Bean
private final Map<String, Set<String>> dependenciesForBeanMap = new ConcurrentHashMap<>(64);

isSingleton

sharedInstance = getSingleton(beanName, () -> {
                        try {
                            return createBean(beanName, mbd, args);
                        }
                        catch (BeansException ex) {
                            // Explicitly remove instance from singleton cache: It might have been put there
                            // eagerly by the creation process, to allow for circular reference resolution.
                            // Also remove any beans that received a temporary reference to the bean.
                            destroySingleton(beanName);
                            throw ex;
                        }
                    });
                    beanInstance = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);

在这个阶段,我们可以看到代码已经在准备创建单例 bean 实例了,因此我们可以不去深入理解这部分的源码逻辑。反正,在 getSingleton 方法中,会调用 createBean 方法。这里使用了 lambda 表达式,如果有不太了解的读者,可以参考下之前发的文章进行学习:

isPrototype

  if (mbd.isPrototype()) {
			// It's a prototype -> create a new instance.
			Object prototypeInstance = null;
			try {
				beforePrototypeCreation(beanName);
				prototypeInstance = createBean(beanName, mbd, args);
			}
			finally {
				afterPrototypeCreation(beanName);
			}
			beanInstance = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
		}

在这个阶段,我们发现创建 bean 的方法已经改变了,直接调用了 createBean 方法,而不是通过 getSingleton 方法进行调用。至于 beforePrototypeCreation 和 afterPrototypeCreation,我们可以不用管它们,因为它们只是存储一些信息,对我们创建 bean 并没有太大的影响。

其他Scope

讲解这部分源码之前,我们先来看看还有哪些Scope域:

//@RequestScope
@SessionScope
public class User {
}

现在我们来看一下 RequestScope 和 SessionScope,它们与其他作用域类似,只是一个组合注解。它们的元注解信息如下:

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Scope(WebApplicationContext.SCOPE_SESSION)
public @interface SessionScope {

	/**
	 * Alias for {@link Scope#proxyMode}.
	 * <p>Defaults to {@link ScopedProxyMode#TARGET_CLASS}.
	 */
	@AliasFor(annotation = Scope.class)
	ScopedProxyMode proxyMode() default ScopedProxyMode.TARGET_CLASS;

}

然后我们再来看下Spring对其他Scope注解的逻辑判断:

String scopeName = mbd.getScope();
					if (!StringUtils.hasLength(scopeName)) {
						throw new IllegalStateException("No scope name defined for bean ´" + beanName + "'");
					}
					Scope scope = this.scopes.get(scopeName);
					if (scope == null) {
						throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
					}
					try {  // session.getAttriute(beaName)  setAttri
						Object scopedInstance = scope.get(beanName, () -> {
							beforePrototypeCreation(beanName);
							try {
								return createBean(beanName, mbd, args);
							}
							finally {
								afterPrototypeCreation(beanName);
							}
						});
						beanInstance = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
					}
					catch (IllegalStateException ex) {
						throw new ScopeNotActiveException(beanName, scopeName, ex);
					}

其实他主要用的getAttriute方法,我们看下scope.get主要的逻辑判断:

	public Object get(String name, ObjectFactory<?> objectFactory) {
		RequestAttributes attributes = RequestContextHolder.currentRequestAttributes();
		Object scopedObject = attributes.getAttribute(name, getScope());
		if (scopedObject == null) {
			scopedObject = objectFactory.getObject();
			attributes.setAttribute(name, scopedObject, getScope());
			// Retrieve object again, registering it for implicit session attribute updates.
			// As a bonus, we also allow for potential decoration at the getAttribute level.
			Object retrievedObject = attributes.getAttribute(name, getScope());
			if (retrievedObject != null) {
				// Only proceed with retrieved object if still present (the expected case).
				// If it disappeared concurrently, we return our locally created instance.
				scopedObject = retrievedObject;
			}
		}
		return scopedObject;
	}

在这个阶段,我们可以看到,通过 objectFactory.getObject() 方法,会调用外层定义的 lambda 表达式,也就是 createBean 方法的逻辑。假设这个过程成功地创建了 bean 实例,并返回了它,那么 Spring 会调用 setAttribute 方法,将这个 bean 实例以及其 scope 值放入以 beanName 为 key 的属性中。这样,当需要获取这个 bean 实例时,Spring 就可以直接从作用域中获取了。

结语

getBean 方法主要包含以下几个步骤:

  1. 首先,从单例缓存池中获取 bean 实例。如果没有,Spring 会创建新的 bean 实例,并将其添加到单例缓存池中。
  2. 接着,Spring 会检查当前容器是否有指定名称的 bean 定义。如果没有,Spring 会调用父容器的 getBean 方法,直到找到为止。
  3. 一旦找到了 bean 定义,Spring 会根据不同的作用域类型,创建对应的 bean 实例,并将其存储在作用域中。
  4. 最后,Spring 会返回创建好的 bean 实例。

非常好,这样我们对 getBean 方法的逻辑判断有了一个大体的了解,有助于我们更好地理解 createBean 方法的实现细节。如果在后续的学习中有任何问题或疑问,可以随时联系我进行咨询。

公众号

原文链接:https://www.cnblogs.com/guoxiaoyu/p/17372572.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Spring源码:Bean生命周期(三) - Python技术站

(0)
上一篇 2023年5月4日
下一篇 2023年5月4日

相关文章

  • Java内存泄漏的原因是什么?

    针对Java内存泄漏原因这一问题,我来详细讲解一下。 什么是Java内存泄漏? 首先,我们需要先了解什么是Java内存泄漏。Java内存泄漏指的是,虽然一些对象已经不再被程序所用,但是Java的垃圾回收器却无法回收这些对象的内存空间。这些没被回收的内存空间被占据,随着时间的推移内存空间将会越来越紧张,最终会导致程序的崩溃。 那么,Java内存泄漏的原因是什么…

    Java 2023年5月11日
    00
  • 讲解Java中的基础类库和语言包的使用

    十分感谢你提出的问题。下面我将详细讲解“讲解Java中的基础类库和语言包的使用”的完整攻略。 什么是Java基础类库和语言包? Java基础类库和语言包是Java语言核心库的一部分,提供了大量的基本类和接口,Java程序都可以直接使用。其中Java基础类库包含很多常用的类和接口,如字符串(String)、集合(Collection)、IO操作(IO)等;而J…

    Java 2023年5月26日
    00
  • win2000/2003下整合IIS+Tomcat5支持jsp

    要在Win2000/2003下整合IIS和Tomcat5来支持JSP,需要按照以下步骤来实现: Step 1. 安装IIS和Tomcat5 首先需要在Windows服务器上安装IIS和Tomcat5。对于IIS,需要在Windows的“控制面板”中选择“添加/删除程序”,然后选择“添加/删除Windows组件”,找到IIS并安装。对于Tomcat5,可以从A…

    Java 2023年5月19日
    00
  • Java hibernate延迟加载get和load的区别

    下面是详细讲解Java Hibernate延迟加载get和load的区别的攻略: 延迟加载的概念 Hibernate是一个开源的ORM(对象关系映射)框架,它提供了对象到关系数据库的映射服务,可以方便地操作数据库。对于大量数据的操作,Hibernate采用了延迟加载的机制,即只有在需要使用数据时才会从数据库中取出数据,以节省内存和网络资源。 Hibernat…

    Java 2023年5月19日
    00
  • Java实现雪花算法的示例代码

    题目:Java实现雪花算法的示例代码 1. 什么是雪花算法? 雪花算法(Snowflake)是Twitter公司开发的一种唯一ID生成算法,它可以生成一个长度为64bit的唯一ID,被广泛应用于分布式系统中,这样可以避免ID冲突的情况。 雪花算法的生成,主要依靠了数据中心ID(5位)、机器ID(5位)、时间戳(41位)以及自增的序列(12位)。 2. 雪花算…

    Java 2023年5月18日
    00
  • JAVA实现监测tomcat是否宕机及控制重启的方法

    下面是详细讲解”JAVA实现监测tomcat是否宕机及控制重启的方法”的完整攻略: 1. 监测Tomcat是否宕机 要监测Tomcat是否宕机,可以使用Java自带的Socket库建立Socket连接来判断Tomcat是否还在运行。下面是示例代码: public class TomcatMonitor { // 定义Tomcat的IP和端口 private …

    Java 2023年6月2日
    00
  • 关于maven环境的安装及maven集成idea环境的问题

    下面是关于maven环境的安装及maven集成idea环境的问题的完整攻略。 1. Maven环境的安装 1.1 下载Maven 首先,需要从Maven官网上下载最新版的Maven。可以访问以下网址: https://maven.apache.org/download.cgi 选择最新版本的二进制zip文件,下载后解压缩到本地。 1.2 配置环境变量 在Ma…

    Java 2023年5月20日
    00
  • Spring Boot 多数据源处理事务的思路详解

    Spring Boot 多数据源处理事务的思路详解 为什么需要多数据源 在实际应用中,我们可能需要连接多个数据库来完成不同的业务需求,例如:用户数据存在 MySQL 数据库中,订单数据存在 MongoDB 数据库中,而且不同的数据源可能有不同的事务管理机制,为了更好地处理多数据源事务,我们需要进行多数据源处理。 Spring Boot 多数据源处理事务方案 …

    Java 2023年6月3日
    00
合作推广
合作推广
分享本页
返回顶部