AbstractApplicationContext#refresh()

  1. @Override
  2. public void refresh() throws BeansException, IllegalStateException {
  3. synchronized (this.startupShutdownMonitor) {
  4. // Prepare this context for refreshing.
  5. // 1. 设置启动时间、开启活跃状态
  6. // 2. 初始化子类自定义属性(placeholder property)
  7. // 3. 验证环境里的属性
  8. prepareRefresh();
  9. // Tell the subclass to refresh the internal bean factory.
  10. // 获取beanFactory,实际类型是DefaultListableBeanFactory
  11. ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
  12. // Prepare the bean factory for use in this context.
  13. // 设置beanFactory内属性,比如类加载器、后置处理器(BeanPostProcessor)
  14. prepareBeanFactory(beanFactory);
  15. try {
  16. // Allows post-processing of the bean factory in context subclasses.
  17. // hook:添加对beanfactory的后置处理配置,此处是个hook,留给子类具体实现(模版方法)
  18. postProcessBeanFactory(beanFactory);
  19. // Invoke factory processors registered as beans in the context.
  20. // 从Spring容器中找出实现了BeanFactoryPostProcessor接口的类并执行。
  21. // 具体执行委托给PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors()
  22. // 执行内容主要是记录注册表以及修改Spring容器内已经存在的BeanDefinition顺便实例化BeanFactoryPostProcessor及其子类BeanDefinitionPostProcessor类型的bean
  23. invokeBeanFactoryPostProcessors(beanFactory);
  24. // Register bean processors that intercept bean creation.
  25. // 1. 获取beanFactory中类型为BeanPostProcessor的BeanName,然后实例化、初始化,保证优先于我们的业务bean
  26. // 2. 装载到beanFactory中(顺序依照priorityOrderedPostProcessors、orderedPostProcessors、internalPostProcessors)
  27. registerBeanPostProcessors(beanFactory);
  28. // Initialize message source for this context.
  29. // 初始化消息源,用于解析消息,支持解析消息参数化和国际化
  30. initMessageSource();
  31. // Initialize event multicaster for this context.
  32. // 初始化ApplicationEventMulticaster(事件广播器),用于事件监听器的注册和事件的广播
  33. // 如果用户没有指定,Spring会默认用SimpleApplicationEventMulticaster作为实现
  34. initApplicationEventMulticaster();
  35. // Initialize other special beans in specific context subclasses.
  36. // hook:单例实例化之前初始化特殊bean,此处实现为空,此处是个hook,子类扩展实现(模版方法)
  37. onRefresh();
  38. // Check for listener beans and register them.
  39. // 将监听器(ApplicationListener)添加到广播器中(ApplicationEventMulticaster)
  40. registerListeners();
  41. // Instantiate all remaining (non-lazy-init) singletons.
  42. // 1. 实例化剩余的单例bean(除了懒加载的实例)
  43. finishBeanFactoryInitialization(beanFactory);
  44. // Last step: publish corresponding event.
  45. finishRefresh();
  46. }
  47. catch (BeansException ex) {
  48. if (logger.isWarnEnabled()) {
  49. logger.warn("Exception encountered during context initialization - " +
  50. "cancelling refresh attempt: " + ex);
  51. }
  52. // Destroy already created singletons to avoid dangling resources.
  53. destroyBeans();
  54. // Reset 'active' flag.
  55. cancelRefresh(ex);
  56. // Propagate exception to caller.
  57. throw ex;
  58. }
  59. finally {
  60. // Reset common introspection caches in Spring's core, since we
  61. // might not ever need metadata for singleton beans anymore...
  62. resetCommonCaches();
  63. }
  64. }
  65. }

#obtainFreshBeanFactory()

获取实际类型是DefaultListableBeanFactoryBeanFactory,并且生成注册BeanDefinitionDefaultListableBeanFactory.beanDefinitionMap容器内。
如果是从ClassPathXmlApplicationContext类进入的话,此处的obtainFreshBeanFactory()是调用其父类AbstractRefreshableApplicationContext的实现。
AbstractRefreshableApplicationContext#obtainFreshBeanFactory()调用的核心方法如下:

  1. @Override
  2. protected final void refreshBeanFactory() throws BeansException {
  3. // 关闭先前的bean工厂(如果存在的话)
  4. if (hasBeanFactory()) {
  5. destroyBeans();
  6. closeBeanFactory();
  7. }
  8. try {
  9. // 初始化一个新工厂
  10. DefaultListableBeanFactory beanFactory = createBeanFactory();
  11. beanFactory.setSerializationId(getId());
  12. // 设置beanFactory的属性,是否允许循环引用、beanDefinition的重写
  13. customizeBeanFactory(beanFactory);
  14. // 1. 解析xml文件转换成document对象,解析节点生成BeanDefinition(解析属性:scope、singleton、class、parent等)
  15. // 2. 注册beanDefinition到DefaultListableBeanFactory.beanDefinitionMap中(spring容器中)
  16. loadBeanDefinitions(beanFactory);
  17. synchronized (this.beanFactoryMonitor) {
  18. this.beanFactory = beanFactory;
  19. }
  20. }
  21. catch (IOException ex) {
  22. throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);
  23. }
  24. }

#invokeBeanFactoryPostProcessors(beanFactory)

修改BeanDefinition对象,方法内关键代码invokeBeanFactoryPostProcessor。如子类ConfigurationClassPostProcessor#postProcessBeanFactory()
�会使用CGLIB对BeanDefinitionBeanClass做代码增强,生成它的CGLIB子类,然后替换掉原来的BeanClass,如下图所示:
image.png

另外在对BeanDefinition对象做增强等修改之前,beanFactory会先根据beanFactory.getBeanNamesForType()方法,将BeanDefinitionRegistryPostProcessorBeanFactoryPostProcessor都先实例化。
简而言之,会实例化BeanFactoryPostProcessorBeanDefinitionRegistryPostProcessor类型的bean,然后使用这些增强bean对BeanDefinition做修改(增强)

#registerBeanPostProcessors(beanFactory)

功能如方法名,实例化并注册BeanPostProcessor。内部是由代理类PostProcessorRegistrationDelegate.registerBeanPostProcessor()实现的,按照优先级分别实例化和注册,优先级队列从高到低分为priorityOrderedPostProcessorsorderedPostProcessorsnoneOrderPostProcessors

#finishBeanFactoryInitialization(beanFactory)

实例化剩余的单例Bean(前面方法已经实例化了BeanFactoryPostProcessorBeanPostProcessor),关键的内部方法是beanFactory.preInstantialSingletons(),该方法内部的核心方法是AbstractBeanFactory#doGetBean()

  1. protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
  2. @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
  3. // 返回工厂名称,比如将别名解析成正规名称
  4. final String beanName = transformedBeanName(name);
  5. Object bean;
  6. // Eagerly check singleton cache for manually registered singletons.
  7. // 从缓存中获取单例bean 对象
  8. Object sharedInstance = getSingleton(beanName);
  9. if (sharedInstance != null && args == null) {
  10. if (logger.isTraceEnabled()) {
  11. if (isSingletonCurrentlyInCreation(beanName)) {
  12. logger.trace("Returning eagerly cached instance of singleton bean '" + beanName +
  13. "' that is not fully initialized yet - a consequence of a circular reference");
  14. }
  15. else {
  16. logger.trace("Returning cached instance of singleton bean '" + beanName + "'");
  17. }
  18. }
  19. bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
  20. }
  21. // 缓存中不存在单例bean 对象
  22. else {
  23. // Fail if we're already creating this bean instance:
  24. // We're assumably within a circular reference.
  25. if (isPrototypeCurrentlyInCreation(beanName)) {
  26. throw new BeanCurrentlyInCreationException(beanName);
  27. }
  28. // Check if bean definition exists in this factory.
  29. // 校验BeanDefinition是否存在Spring容器中
  30. BeanFactory parentBeanFactory = getParentBeanFactory();
  31. if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
  32. // Not found -> check parent.
  33. String nameToLookup = originalBeanName(name);
  34. if (parentBeanFactory instanceof AbstractBeanFactory) {
  35. return ((AbstractBeanFactory) parentBeanFactory).doGetBean(
  36. nameToLookup, requiredType, args, typeCheckOnly);
  37. }
  38. else if (args != null) {
  39. // Delegation to parent with explicit args.
  40. return (T) parentBeanFactory.getBean(nameToLookup, args);
  41. }
  42. else if (requiredType != null) {
  43. // No args -> delegate to standard getBean method.
  44. return parentBeanFactory.getBean(nameToLookup, requiredType);
  45. }
  46. else {
  47. return (T) parentBeanFactory.getBean(nameToLookup);
  48. }
  49. }
  50. if (!typeCheckOnly) {
  51. markBeanAsCreated(beanName);
  52. }
  53. try {
  54. // 获取BeanDefinition对象
  55. final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
  56. checkMergedBeanDefinition(mbd, beanName, args);
  57. // Guarantee initialization of beans that the current bean depends on.
  58. // 获取依赖bean对象,保证依赖bean对象先被创建
  59. String[] dependsOn = mbd.getDependsOn();
  60. if (dependsOn != null) {
  61. for (String dep : dependsOn) {
  62. if (isDependent(beanName, dep)) {
  63. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  64. "Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
  65. }
  66. registerDependentBean(dep, beanName);
  67. try {
  68. getBean(dep);
  69. }
  70. catch (NoSuchBeanDefinitionException ex) {
  71. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  72. "'" + beanName + "' depends on missing bean '" + dep + "'", ex);
  73. }
  74. }
  75. }
  76. // Create bean instance.
  77. // 如果是单例(scope为singleton),则创建对象
  78. if (mbd.isSingleton()) {
  79. sharedInstance = getSingleton(beanName, () -> {
  80. try {
  81. return createBean(beanName, mbd, args);
  82. }
  83. catch (BeansException ex) {
  84. // Explicitly remove instance from singleton cache: It might have been put there
  85. // eagerly by the creation process, to allow for circular reference resolution.
  86. // Also remove any beans that received a temporary reference to the bean.
  87. destroySingleton(beanName);
  88. throw ex;
  89. }
  90. });
  91. bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
  92. }
  93. // 创建scope为prototype(每次调用都会创建一个实例)对象
  94. else if (mbd.isPrototype()) {
  95. // It's a prototype -> create a new instance.
  96. Object prototypeInstance = null;
  97. try {
  98. beforePrototypeCreation(beanName);
  99. prototypeInstance = createBean(beanName, mbd, args);
  100. }
  101. finally {
  102. afterPrototypeCreation(beanName);
  103. }
  104. bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
  105. }
  106. else {
  107. String scopeName = mbd.getScope();
  108. final Scope scope = this.scopes.get(scopeName);
  109. if (scope == null) {
  110. throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
  111. }
  112. try {
  113. Object scopedInstance = scope.get(beanName, () -> {
  114. beforePrototypeCreation(beanName);
  115. try {
  116. return createBean(beanName, mbd, args);
  117. }
  118. finally {
  119. afterPrototypeCreation(beanName);
  120. }
  121. });
  122. bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
  123. }
  124. catch (IllegalStateException ex) {
  125. throw new BeanCreationException(beanName,
  126. "Scope '" + scopeName + "' is not active for the current thread; consider " +
  127. "defining a scoped proxy for this bean if you intend to refer to it from a singleton",
  128. ex);
  129. }
  130. }
  131. }
  132. catch (BeansException ex) {
  133. cleanupAfterBeanCreationFailure(beanName);
  134. throw ex;
  135. }
  136. }
  137. // Check if required type matches the type of the actual bean instance.
  138. if (requiredType != null && !requiredType.isInstance(bean)) {
  139. try {
  140. T convertedBean = getTypeConverter().convertIfNecessary(bean, requiredType);
  141. if (convertedBean == null) {
  142. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  143. }
  144. return convertedBean;
  145. }
  146. catch (TypeMismatchException ex) {
  147. if (logger.isTraceEnabled()) {
  148. logger.trace("Failed to convert bean '" + name + "' to required type '" +
  149. ClassUtils.getQualifiedName(requiredType) + "'", ex);
  150. }
  151. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  152. }
  153. }
  154. return (T) bean;
  155. }

方法内部调用了getSingleton(),具体方法如下:

  1. protected Object getSingleton(String beanName, boolean allowEarlyReference) {
  2. // 一级缓存 singletonObjects
  3. Object singletonObject = this.singletonObjects.get(beanName);
  4. if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
  5. synchronized (this.singletonObjects) {
  6. // 二级缓存 earlySingletonObjects
  7. singletonObject = this.earlySingletonObjects.get(beanName);
  8. if (singletonObject == null && allowEarlyReference) {
  9. // 三级缓存 singletonFactories
  10. ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
  11. if (singletonFactory != null) {
  12. // 通过ObjectFactory#getObject()创建
  13. singletonObject = singletonFactory.getObject();
  14. // 创建成功放入二级缓存
  15. this.earlySingletonObjects.put(beanName, singletonObject);
  16. this.singletonFactories.remove(beanName);
  17. }
  18. }
  19. }
  20. }
  21. return singletonObject;
  22. }

缓存中不存在则创建bean,createBean(beanName, mbd, args),内部逻辑是先给BeanPostProcessor一个机会扫描下需要创建的bean(resolveBeforeInstantiation(beanName, mbdToUse)
�),如果bean符合条件目标类型生成代理类直接返回
image.png
resolveBeforeInstantiation(beanName, mbdToUse)代码如下,通过Object result = ibp.postProcessBeforeInstantiation(beanClass, beanName);InstantiationAwareBeanPostProcessor的子类创建了代理类。

  1. protected Object resolveBeforeInstantiation(String beanName, RootBeanDefinition mbd) {
  2. Object bean = null;
  3. if (!Boolean.FALSE.equals(mbd.beforeInstantiationResolved)) {
  4. // Make sure bean class is actually resolved at this point.
  5. if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
  6. Class<?> targetType = determineTargetType(beanName, mbd);
  7. if (targetType != null) {
  8. bean = applyBeanPostProcessorsBeforeInstantiation(targetType, beanName);
  9. if (bean != null) {
  10. bean = applyBeanPostProcessorsAfterInitialization(bean, beanName);
  11. }
  12. }
  13. }
  14. mbd.beforeInstantiationResolved = (bean != null);
  15. }
  16. return bean;
  17. }
  18. protected Object applyBeanPostProcessorsBeforeInstantiation(Class<?> beanClass, String beanName) {
  19. for (BeanPostProcessor bp : getBeanPostProcessors()) {
  20. if (bp instanceof InstantiationAwareBeanPostProcessor) {
  21. InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
  22. // 创建代理类
  23. Object result = ibp.postProcessBeforeInstantiation(beanClass, beanName);
  24. if (result != null) {
  25. return result;
  26. }
  27. }
  28. }
  29. return null;
  30. }


createBean(beanName, mbd, args)其内部另个核心方法是doCreateBean()具体方法如下,简而言之就是:实例化bean—放入三级缓存—填充属性-初始化

  1. protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
  2. throws BeanCreationException {
  3. // Instantiate the bean.
  4. BeanWrapper instanceWrapper = null;
  5. if (mbd.isSingleton()) {
  6. instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
  7. }
  8. if (instanceWrapper == null) {
  9. // 创建bean实例
  10. instanceWrapper = createBeanInstance(beanName, mbd, args);
  11. }
  12. final Object bean = instanceWrapper.getWrappedInstance();
  13. Class<?> beanType = instanceWrapper.getWrappedClass();
  14. if (beanType != NullBean.class) {
  15. mbd.resolvedTargetType = beanType;
  16. }
  17. // Allow post-processors to modify the merged bean definition.
  18. synchronized (mbd.postProcessingLock) {
  19. if (!mbd.postProcessed) {
  20. try {
  21. applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
  22. }
  23. catch (Throwable ex) {
  24. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  25. "Post-processing of merged bean definition failed", ex);
  26. }
  27. mbd.postProcessed = true;
  28. }
  29. }
  30. // Eagerly cache singletons to be able to resolve circular references
  31. // even when triggered by lifecycle interfaces like BeanFactoryAware.
  32. boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
  33. isSingletonCurrentlyInCreation(beanName));
  34. if (earlySingletonExposure) {
  35. if (logger.isTraceEnabled()) {
  36. logger.trace("Eagerly caching bean '" + beanName +
  37. "' to allow for resolving potential circular references");
  38. }
  39. // 加入到三级缓存
  40. addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
  41. }
  42. // Initialize the bean instance.
  43. Object exposedObject = bean;
  44. try {
  45. // 填充属性
  46. populateBean(beanName, mbd, instanceWrapper);
  47. // 初始化
  48. exposedObject = initializeBean(beanName, exposedObject, mbd);
  49. }
  50. catch (Throwable ex) {
  51. if (ex instanceof BeanCreationException && beanName.equals(((BeanCreationException) ex).getBeanName())) {
  52. throw (BeanCreationException) ex;
  53. }
  54. else {
  55. throw new BeanCreationException(
  56. mbd.getResourceDescription(), beanName, "Initialization of bean failed", ex);
  57. }
  58. }
  59. if (earlySingletonExposure) {
  60. Object earlySingletonReference = getSingleton(beanName, false);
  61. if (earlySingletonReference != null) {
  62. if (exposedObject == bean) {
  63. exposedObject = earlySingletonReference;
  64. }
  65. else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
  66. String[] dependentBeans = getDependentBeans(beanName);
  67. Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
  68. for (String dependentBean : dependentBeans) {
  69. if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
  70. actualDependentBeans.add(dependentBean);
  71. }
  72. }
  73. if (!actualDependentBeans.isEmpty()) {
  74. throw new BeanCurrentlyInCreationException(beanName,
  75. "Bean with name '" + beanName + "' has been injected into other beans [" +
  76. StringUtils.collectionToCommaDelimitedString(actualDependentBeans) +
  77. "] in its raw version as part of a circular reference, but has eventually been " +
  78. "wrapped. This means that said other beans do not use the final version of the " +
  79. "bean. This is often the result of over-eager type matching - consider using " +
  80. "'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.");
  81. }
  82. }
  83. }
  84. }
  85. // Register bean as disposable.
  86. try {
  87. registerDisposableBeanIfNecessary(beanName, bean, mbd);
  88. }
  89. catch (BeanDefinitionValidationException ex) {
  90. throw new BeanCreationException(
  91. mbd.getResourceDescription(), beanName, "Invalid destruction signature", ex);
  92. }
  93. return exposedObject;
  94. }

初始化方法initialBean():【4】【5】

  1. protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {
  2. if (System.getSecurityManager() != null) {
  3. AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
  4. invokeAwareMethods(beanName, bean);
  5. return null;
  6. }, getAccessControlContext());
  7. }
  8. else {
  9. // 执行实现了Aware接口中的方法
  10. invokeAwareMethods(beanName, bean);
  11. }
  12. Object wrappedBean = bean;
  13. if (mbd == null || !mbd.isSynthetic()) {
  14. //调用BeanPostProcessor的postProcessorsBeforeInitialization方法
  15. wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
  16. }
  17. try {
  18. // 调用初始化方法(自定义的初始化方法或者实现InitialzingBean接口)
  19. invokeInitMethods(beanName, wrappedBean, mbd);
  20. }
  21. catch (Throwable ex) {
  22. throw new BeanCreationException(
  23. (mbd != null ? mbd.getResourceDescription() : null),
  24. beanName, "Invocation of init method failed", ex);
  25. }
  26. if (mbd == null || !mbd.isSynthetic()) {
  27. // 调用BeanPostProcessor的postProcessAfterInitialization方法
  28. wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
  29. }
  30. return wrappedBean;
  31. }

applyBeanPostProcessorsAfterInitialization创建代理类:
image.png

关键类梳理

DefalutListableBeanFactory

image.png

AbstractApplicationContext

image.png

BeanPostProcessor【1】

image.png

事件监听器(ApplicationEvent)和广播器(ApplicationEventMulticaster)【2】

Spring的事件监听是一种观察者模式,只要是观察者模式一定包含主题(Subject)、订阅者(Observer)、发布人(Publisher)。
Spring的事件机制主要包括4个类:

  • ApplicationEvent:消息事件,每个实现类代表一个具体事件(主题)
  • ApplicationListener:事件监听器,处理接收到的事件(订阅者)
  • ApplicationEventMulticaster:事件广播器,用于事件监听器的注册和事件的广播
  • ApplicationEventPublisher:事件发布者,委托ApplicationEventMulticatser完成事件发布(发布人)

    事件广播器

    事件广播器,Spring默认实现是SimpleApplicationMulticaster作为广播器。

    事件发布者

    实现ApplicationEventPublisher的即为事件发布者,Spring中的ApplicationContext实现了ApplicationEventPublisher接口,所以应用上下文本身就是一个事件发布者。

参考

【1】别再问Spring Bean的生命周期了!:https://mp.weixin.qq.com/s?src=11&timestamp=1633596915&ver=3359&signature=PGv-XuqtF8XEo5qctBzabuXNenhtLjOCMR07nhqa-aHAaS9MbjAaogGZ5IqMZNzgZ3ylh41r1PDvtRi8Ox8YBkoUQcVCZP4xrQ50HL*qld0isnrjD51B9uKOk44rFM&new=1
【2】Spring Event 事件发布/监听机制 详解并使用:https://mp.weixin.qq.com/s?src=11&timestamp=1633613866&ver=3360&signature=EOBrJwuu2D9RV5uPXRSW8MelYqy0jOS97F8gSRqdtbsEh89DS29sHWXipzfdhzgGtvuIiiHXH8zVh-n8i-zHV6w7VA-m*qImo0p3gC3uAnT5TH7RqVr7kxJMe97Z6nV6&new=1
【3】《轻松读懂spring》之 IOC的主干流程(上):https://mp.weixin.qq.com/s?src=11&timestamp=1633181508&ver=3350&signature=fdRcdb8clLPSBWTzFUSF8loeg6pMyNhUQYAmxIt-GdNoDqzumMb0jBM6Zy8g-B7VfVYGr0Yd2Spm7M-w9VnAb-98VZSaAdjP9UrDBXDRzqZ0iuOE0tuGeKqzSZdr2k&new=1
【4】Spring的Bean的生命周期_面试系列之Spring :https://mp.weixin.qq.com/s?src=11&timestamp=1633684945&ver=3362&signature=D3sYy9hoBtylApU8BWW9SoycqhbevViO3QLKNDEy1vQxs9BXn76Gfu39kjJmVKDy4ttrWz-BbWzrt2GXHaPAITBiJlwGGSYV6oaRFzqpN-o6FdbCHmakA6OAZOtsxk&new=1
【5】spring 依赖注入时,什么时候会创建代理类:https://www.yuque.com/u1687194/hspkbf/lcqlhv#lJ34A