@Scope注解 、@Lazy注解 -【Spring底层原理】

向右看齐 2021-09-21 17:32 845阅读 0赞

目录

一、注解用法

二、实例分析

三、源码追踪

四、总结


一、注解用法

【1】@Scope注解

@Scope注解是用来控制实例作用域的,单实例还是多实例,该注解可以作用在类和方法上面,通过属性来控制作用域,如下:

  • prototype:多实例,IOC容器启动的时候并不会创建对象放在容器中,每次获取的时候才会调用方法创建对象
  • singleton:单实例,IOC容器启动的时候就会调用方法创建对象放到容器中,以后每次获取都是从容器map中拿同一个对象
  • request:同一次请求创建一个实例
  • session:同一个session创建一个实例

【2】@Lazy注解

@Lazy注解,懒加载:即容器启动不创建对象,第一次使用(获取)Bean创建对象,并初始化。

一般的单实例Bean,默认在容器启动的时候创建对象,因此可以使用@Lazy注解改变这一特性,通过加上@Lazy注解,让容器启动不创建对象,第一次使用的时候再创建Bean对象。

二、实例分析

我们可以通过实例来分析证明上面注解的用法,这里有两个问题需要验证:

  1. 单例还是多例
  2. 何时创建的对象(容器启动的时候创建还是方法调用的时候创建)

同样以spring工程为例,咱们来进行验证

【1】单例还是多例

这里使用applicationContext来获取对象,按照id来获取,获取之后进行对比,看是否是同一个对象

  1. // 启动类
  2. public class MainTest {
  3. @Test
  4. public void TestMain(){
  5. AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);
  6. // 按照id获取bean
  7. Object user = applicationContext.getBean("user");
  8. Object user2 = applicationContext.getBean("user");
  9. System.out.println(user == user2);
  10. }
  11. }
  12. // 配置类
  13. @Configuration
  14. public class AppConfig {
  15. // 默认是单例的
  16. // @Scope("prototype"):多实例,IOC容器启动的时候并不会创建对象放在容器中,每次获取的时候才会调用方法创建对象
  17. // @Scope("singleton"):单实例,IOC容器启动的时候就会调用方法创建对象放到容器中,以后每次获取都是从容器map中拿同一个对象
  18. // @Scope("request"):同一次请求创建一个实例
  19. // @Scope("session"):同一个session创建一个实例
  20. @Scope("singleton")
  21. @Bean
  22. // @Lazy
  23. public User user(){
  24. return new User();
  25. }
  26. }
  27. // User的bean
  28. public class User {
  29. public User() {
  30. System.out.println("User对象");
  31. }
  32. }

运行启动类,可以看到user == suer2是true,也就是同一个对象,这是@Scope("singleton")注解的作用,当然,默认情况也是这样

image-20210131235826642

接下来,咱们改为@Scope("prototype")注解,也就是多实例模式,运行启动类,可以看到,返回值是false,也就是说两次创建的对象不是同一个,是多例的(其实这里也可以看到,两次打印,上面只创建了一个User对象,而这里创建了两个User对象,这里就已经能说明单例和多例模式下何时创建对象的了,下面还会进行发分析)

image-20210201000858114

【2】对象何时创建

上面我们提到,单例模式是下容器启动的时候创建的对象,而多例模式是在方法调用的时候创建的对象,所以第一次创建一个,第二次创建两个,这也证实了上面的结论。为了更直观的查看,我们对案例进行如下修改

@Scope注解

我们先将启动类的获取Bean方法注释掉,只留下容器启动的方法,然后运行,

单例模式下(@Scope("singleton")):可以看到,创建了User对象

image-20210201001654209

多例子模式下(@Scope("prototype")):可以看到,没有创建对象

image-20210201001826342

因为没有了方法调用,所以只有在容器启动的时候才能创建对象,所以单例模式是在容器启动的时候创建对象的。

@Lazy注解

单例模式下,是可以在容器启动的时候创建对象的,而使用了@Lazy懒加载注解后,可以改变这一特性,让对象在方法调用的时候再创建

加上@Lazy注解,打印结果如下,可以看到并没有创建对象:

image-20210201003837395

放开启动类的获取Bean方法注释,咱们再次运行,可以看到创建了一个对象:

image-20210201003956197

这也证明了:通过加上@Lazy注解,让容器启动不创建对象,第一次使用的时候再创建Bean对象。

三、源码追踪

这里对单例和多例实例化过程底层原理进行源码追踪

咱们直接来到BeanFactoryPostProcessor接口,找到和Scope注解相关的实现类,进入源码查看:

image-20210201003956197

在这里,调用registerScope方法,继续跟踪

  1. public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
  2. if (this.scopes != null) {
  3. this.scopes.forEach((scopeKey, value) -> {
  4. if (value instanceof Scope) {
  5. beanFactory.registerScope(scopeKey, (Scope)value);
  6. } else {
  7. Class scopeClass;
  8. if (value instanceof Class) {
  9. scopeClass = (Class)value;
  10. Assert.isAssignable(Scope.class, scopeClass, "Invalid scope class");
  11. beanFactory.registerScope(scopeKey, (Scope)BeanUtils.instantiateClass(scopeClass));
  12. } else {
  13. if (!(value instanceof String)) {
  14. throw new IllegalArgumentException("Mapped value [" + value + "] for scope key [" + scopeKey + "] is not an instance of required type [" + Scope.class.getName() + "] or a corresponding Class or String value indicating a Scope implementation");
  15. }
  16. scopeClass = ClassUtils.resolveClassName((String)value, this.beanClassLoader);
  17. Assert.isAssignable(Scope.class, scopeClass, "Invalid scope class");
  18. beanFactory.registerScope(scopeKey, (Scope)BeanUtils.instantiateClass(scopeClass));
  19. }
  20. }
  21. });
  22. }
  23. }

这里是进行注册Scope,在该类中,有个doGetBean方法,所有的bean的创建都会去调用该方法,咱们继续跟踪源码,看Scope是如何创建的:

  1. public void registerScope(String scopeName, Scope scope) {
  2. Assert.notNull(scopeName, "Scope identifier must not be null");
  3. Assert.notNull(scope, "Scope must not be null");
  4. if (!"singleton".equals(scopeName) && !"prototype".equals(scopeName)) {
  5. Scope previous = (Scope)this.scopes.put(scopeName, scope);
  6. if (previous != null && previous != scope) {
  7. if (this.logger.isDebugEnabled()) {
  8. this.logger.debug("Replacing scope '" + scopeName + "' from [" + previous + "] to [" + scope + "]");
  9. }
  10. } else if (this.logger.isTraceEnabled()) {
  11. this.logger.trace("Registering scope '" + scopeName + "' with implementation [" + scope + "]");
  12. }
  13. } else {
  14. throw new IllegalArgumentException("Cannot replace existing scopes 'singleton' and 'prototype'");
  15. }
  16. }

AbstractBeanFactory类中doGetBean方法:

  1. protected <T> T doGetBean(String name, @Nullable Class<T> requiredType, @Nullable Object[] args, boolean typeCheckOnly) throws BeansException {
  2. // 转化传入的Bean,如果是别名转真名,如果是&开头,去掉&。&代表取factoryBean
  3. String beanName = this.transformedBeanName(name);
  4. // 1、这里为了解决Singleton循环依赖,如果允许早期暴露,会在这里取到其早期暴露,后面会介绍
  5. // 2、如果是工厂模式,在获取bean的时候,在这里会先获取到factoryBean
  6. Object sharedInstance = this.getSingleton(beanName);
  7. Object bean;
  8. // 如果 sharedInstance 不为空,说明当前singleton是依赖者在注入属性时创建的。则打印相关日志
  9. if (sharedInstance != null && args == null) {
  10. if (this.logger.isTraceEnabled()) {
  11. if (this.isSingletonCurrentlyInCreation(beanName)) {
  12. this.logger.trace("Returning eagerly cached instance of singleton bean '" + beanName + "' that is not fully initialized yet - a consequence of a circular reference");
  13. } else {
  14. this.logger.trace("Returning cached instance of singleton bean '" + beanName + "'");
  15. }
  16. }
  17. // 这里会将name和beanName同时传进来, 因为一开始我们的name可能是带&符号的,代表这里需要获取factoryBean而不是bean
  18. bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, (RootBeanDefinition)null);
  19. } else {
  20. if (this.isPrototypeCurrentlyInCreation(beanName)) {
  21. throw new BeanCurrentlyInCreationException(beanName);
  22. }
  23. // 判断是否有父容器,如果有父容器,优先用父容器调用其getBean
  24. BeanFactory parentBeanFactory = this.getParentBeanFactory();
  25. if (parentBeanFactory != null && !this.containsBeanDefinition(beanName)) {
  26. String nameToLookup = this.originalBeanName(name);
  27. if (parentBeanFactory instanceof AbstractBeanFactory) {
  28. return ((AbstractBeanFactory)parentBeanFactory).doGetBean(nameToLookup, requiredType, args, typeCheckOnly);
  29. }
  30. if (args != null) {
  31. return parentBeanFactory.getBean(nameToLookup, args);
  32. }
  33. if (requiredType != null) {
  34. return parentBeanFactory.getBean(nameToLookup, requiredType);
  35. }
  36. return parentBeanFactory.getBean(nameToLookup);
  37. }
  38. if (!typeCheckOnly) {
  39. this.markBeanAsCreated(beanName);
  40. }
  41. StartupStep beanCreation = this.applicationStartup.start("spring.beans.instantiate").tag("beanName", name);
  42. try {
  43. if (requiredType != null) {
  44. beanCreation.tag("beanType", requiredType::toString);
  45. }
  46. // 获取目标bean的definition
  47. RootBeanDefinition mbd = this.getMergedLocalBeanDefinition(beanName);
  48. this.checkMergedBeanDefinition(mbd, beanName, args);
  49. // 实例化依赖bean
  50. String[] dependsOn = mbd.getDependsOn();
  51. String[] var12;
  52. if (dependsOn != null) {
  53. var12 = dependsOn;
  54. int var13 = dependsOn.length;
  55. for(int var14 = 0; var14 < var13; ++var14) {
  56. String dep = var12[var14];
  57. if (this.isDependent(beanName, dep)) {
  58. throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
  59. }
  60. this.registerDependentBean(dep, beanName);
  61. try {
  62. this.getBean(dep);
  63. } catch (NoSuchBeanDefinitionException var33) {
  64. throw new BeanCreationException(mbd.getResourceDescription(), beanName, "'" + beanName + "' depends on missing bean '" + dep + "'", var33);
  65. }
  66. }
  67. }
  68. // 这里就是和单例相关的,开始创建Singleton
  69. if (mbd.isSingleton()) {
  70. // 这里用了一个实现ObjectFactory接口 的 lambda表达式。getSingleton会有对这个createBean,有个前置操作和后置操作,中间调用lambda进行创建
  71. sharedInstance = this.getSingleton(beanName, () -> {
  72. try {
  73. return this.createBean(beanName, mbd, args);
  74. } catch (BeansException var5) {
  75. this.destroySingleton(beanName);
  76. throw var5;
  77. }
  78. });
  79. bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
  80. }
  81. // 多例相关,创建Prototype
  82. else if (mbd.isPrototype()) {
  83. var12 = null;
  84. Object prototypeInstance;
  85. try {
  86. this.beforePrototypeCreation(beanName);
  87. prototypeInstance = this.createBean(beanName, mbd, args);
  88. } finally {
  89. this.afterPrototypeCreation(beanName);
  90. }
  91. bean = this.getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
  92. } else {
  93. String scopeName = mbd.getScope();
  94. if (!StringUtils.hasLength(scopeName)) {
  95. throw new IllegalStateException("No scope name defined for bean ��" + beanName + "'");
  96. }
  97. Scope scope = (Scope)this.scopes.get(scopeName);
  98. if (scope == null) {
  99. throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
  100. }
  101. try {
  102. // scope创建和single差不多, 也是调用get by beanName + lambda表达式, 这个scope会对scopeBean生命进行管理
  103. Object scopedInstance = scope.get(beanName, () -> {
  104. // 也有before, 只是和singleton有点不一样, 它的before和after封装在lambda中
  105. this.beforePrototypeCreation(beanName);
  106. Object var4;
  107. try {
  108. var4 = this.createBean(beanName, mbd, args);
  109. } finally {
  110. this.afterPrototypeCreation(beanName);
  111. }
  112. return var4;
  113. });
  114. bean = this.getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
  115. } catch (IllegalStateException var32) {
  116. throw new ScopeNotActiveException(beanName, scopeName, var32);
  117. }
  118. }
  119. } catch (BeansException var35) {
  120. beanCreation.tag("exception", var35.getClass().toString());
  121. beanCreation.tag("message", String.valueOf(var35.getMessage()));
  122. this.cleanupAfterBeanCreationFailure(beanName);
  123. throw var35;
  124. } finally {
  125. beanCreation.end();
  126. }
  127. }
  128. if (requiredType != null && !requiredType.isInstance(bean)) {
  129. try {
  130. T convertedBean = this.getTypeConverter().convertIfNecessary(bean, requiredType);
  131. if (convertedBean == null) {
  132. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  133. } else {
  134. return convertedBean;
  135. }
  136. } catch (TypeMismatchException var34) {
  137. if (this.logger.isTraceEnabled()) {
  138. this.logger.trace("Failed to convert bean '" + name + "' to required type '" + ClassUtils.getQualifiedName(requiredType) + "'", var34);
  139. }
  140. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  141. }
  142. } else {
  143. return bean;
  144. }
  145. }

这个方法比较长,内容比较多,咱们挑重要的分开来进行讲解:

【1】获取单例对象

  1. if (sharedInstance != null && args == null) {
  2. if (this.logger.isTraceEnabled()) {
  3. if (this.isSingletonCurrentlyInCreation(beanName)) {
  4. this.logger.trace("Returning eagerly cached instance of singleton bean '" + beanName + "' that is not fully initialized yet - a consequence of a circular reference");
  5. } else {
  6. this.logger.trace("Returning cached instance of singleton bean '" + beanName + "'");
  7. }
  8. }
  9. bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, (RootBeanDefinition)null);
  10. } else {
  11. // 创建对象
  12. }

首先,会去spring的容器里面根据名字查找单例,如果获取到单例实例对象,就直接返回该实例,如果获取不到,则进入到else分支,就是创建单例,创建单例之前,首先会把目标bean里面的所有依赖bean(并非@Autowired注解的bean,而是xml用到depend-on标签,或者是参数值引用的bean)全都创建完才创建目标bean,部分代码如下:

【2】实例化依赖bean

  1. // 获取目标bean的definition
  2. RootBeanDefinition mbd = this.getMergedLocalBeanDefinition(beanName);
  3. this.checkMergedBeanDefinition(mbd, beanName, args);
  4. // 实例化依赖bean
  5. String[] dependsOn = mbd.getDependsOn();
  6. String[] var12;
  7. if (dependsOn != null) {
  8. var12 = dependsOn;
  9. int var13 = dependsOn.length;
  10. for(int var14 = 0; var14 < var13; ++var14) {
  11. String dep = var12[var14];
  12. if (this.isDependent(beanName, dep)) {
  13. throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
  14. }
  15. this.registerDependentBean(dep, beanName);
  16. try {
  17. this.getBean(dep);
  18. } catch (NoSuchBeanDefinitionException var33) {
  19. throw new BeanCreationException(mbd.getResourceDescription(), beanName, "'" + beanName + "' depends on missing bean '" + dep + "'", var33);
  20. }
  21. }
  22. }

从源码可以看出,依赖bean会在目标bean之前被创建出来。创建完依赖bean之后,就开始创建目标bean,首先他会先判断bean的定义是否是单例,如果是,会先去缓存查找是否已经创建了单例,如果创建了直接返回,否则创建单例并返回;如果不是则创建多例,创建单例的部分代码如下:

【3】创建单例对象

  1. if (mbd.isSingleton())
  2. {
  3. sharedInstance = this.getSingleton(beanName, () -> {
  4. try {
  5. return this.createBean(beanName, mbd, args);
  6. } catch (BeansException var5) {
  7. this.destroySingleton(beanName);
  8. throw var5;
  9. }
  10. });
  11. bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
  12. }

如果是多例则会进入到多例的创建流程,部分代码如下:

【4】创建多例对象

  1. if (mbd.isPrototype()) {
  2. var12 = null;
  3. Object prototypeInstance;
  4. try {
  5. this.beforePrototypeCreation(beanName);
  6. prototypeInstance = this.createBean(beanName, mbd, args);
  7. } finally {
  8. this.afterPrototypeCreation(beanName);
  9. }
  10. bean = this.getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
  11. }

单例和多例创建的差异:

  • 单例创建完后spring会记录他的实例状态并保存到缓存里面,下次创建会优先取缓存里的实例
  • 多例的创建则是直接创建实例化对象,并不会记录他的实例

在上面的代码里面,bean = this.getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);出现的比较频繁,我们知道,spring里面有一种叫做bean工厂的,就是实现了接口FactoryBean的类,他的实例并不是真正的bean,他是一特定类的创建的工厂类,所以工厂的bean实例化完后并不能直接返回,bean工厂的具体的实例化对象是通过接口FactoryBean里面的方法getObject来实现的,所以如果是工厂类则需要特殊处理,这也是正是这句代码的作用,我们来看下它的实现:

  1. protected Object getObjectForBeanInstance(Object beanInstance, String name, String beanName, @Nullable RootBeanDefinition mbd) {
  2. if (BeanFactoryUtils.isFactoryDereference(name)) {
  3. if (beanInstance instanceof NullBean) {
  4. return beanInstance;
  5. } else if (!(beanInstance instanceof FactoryBean)) {
  6. throw new BeanIsNotAFactoryException(beanName, beanInstance.getClass());
  7. } else {
  8. if (mbd != null) {
  9. mbd.isFactoryBean = true;
  10. }
  11. return beanInstance;
  12. }
  13. } else if (!(beanInstance instanceof FactoryBean)) {
  14. return beanInstance;
  15. } else {
  16. Object object = null;
  17. if (mbd != null) {
  18. mbd.isFactoryBean = true;
  19. } else {
  20. object = this.getCachedObjectForFactoryBean(beanName);
  21. }
  22. if (object == null) {
  23. FactoryBean<?> factory = (FactoryBean)beanInstance;
  24. if (mbd == null && this.containsBeanDefinition(beanName)) {
  25. mbd = this.getMergedLocalBeanDefinition(beanName);
  26. }
  27. boolean synthetic = mbd != null && mbd.isSynthetic();
  28. object = this.getObjectFromFactoryBean(factory, beanName, !synthetic);
  29. }
  30. return object;
  31. }
  32. }

从代码里我们可以看出,要是如果是个普通的bean则就会直接返回实例对象,如果是个工厂bean,则会去执行获取工厂里的实例化对象。

四、总结

主要的实现都是在doGetBean方法中进行的

  1. doGetBean方法中,通过getSingleton(beanName)先去缓存中根据名字获取单例对象,获取到则直接返回
  2. 如果获取不到就创建实例对象,创建之前会先创建依赖Bean,创建完依赖Bean后就开始创建目标Bean
  3. 创建的时候会判断bean的定义是否是单例
  4. 是单例==>去缓存查找是否已经创建了单例,如果创建了直接返回
  5. 不是单例==>则创建多例

发表评论

表情:
评论列表 (有 0 条评论,845人围观)

还没有评论,来说两句吧...

相关阅读