Spring在BeanFactory基礎上提供了一些列具體容器的實現,其中AnnotationConfigApplicationContext是一個用來管理注解bean的容器,從AnnotationConfigApplicationContext的實現結構圖中可以看出:
- AnnotationConfigApplicationContext繼承GenericApplicationContext這個通用應用上下文,GenericApplicationContext內部定義了一個DefaultListableBeanFactory實例,GenericApplicationContext實現了BeanDefinitionRegistry接口,所以可以通過AnnotationConfigApplicationContext實例注冊bean defintion,然后調用refresh()方法來初始化上下文。
- AnnotationConfigApplicationContext繼承AbstractApplicationContext,AbstractApplicationContext提供了ApplicationContext的抽象實現。
下面通過一個示例分析AnnotationConfigApplicationContext的初始化過程:
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(ExtensionConfig.class);
構造函數:
1 //1. 初始化bean讀取器和掃描器; 2 //調用父類GenericApplicationContext無參構造函數,初始化一個BeanFactory: DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory() 3 this(); 4 //2.注冊bean配置類 5 register(annotatedClasses); 6 //3.刷新上下文 7 refresh(); 8 }
1. this() 初始化bean讀取器和掃描器
1 public AnnotationConfigApplicationContext() { 2 //在IOC容器中初始化一個 注解bean讀取器AnnotatedBeanDefinitionReader 3 this.reader = new AnnotatedBeanDefinitionReader(this); 4 //在IOC容器中初始化一個 按類路徑掃描注解bean的 掃描器 5 this.scanner = new ClassPathBeanDefinitionScanner(this);
父類GenericApplicationContext部分代碼:
1 public class GenericApplicationContext extends AbstractApplicationContext implements BeanDefinitionRegistry { 2 private final DefaultListableBeanFactory beanFactory; 3 4 //初始化一個BeanFactory 5 public GenericApplicationContext() { 6 this.beanFactory = new DefaultListableBeanFactory(); 7 } 8 9 … 10 }
2. register(annotatedClasses)
注冊bean配置類, AnnotationConfigApplicationContext容器通過AnnotatedBeanDefinitionReader的register方法實現注解bean的讀取,具體源碼如下:
AnnotationConfigApplicationContext.java中register方法
1 //按指定bean配置類讀取bean 2 public void register(Class<?>... annotatedClasses) { 3 for (Class<?> annotatedClass : annotatedClasses) { 4 registerBean(annotatedClass); 5 } 6 } 7 8 public void registerBean(Class<?> annotatedClass) { 9 doRegisterBean(annotatedClass, null, null, null); 10 } 11 12 //核心實現邏輯 13 <T> void doRegisterBean(Class<T> annotatedClass, @Nullable Supplier<T> instanceSupplier, @Nullable String name, 14 @Nullable Class<? extends Annotation>[] qualifiers, BeanDefinitionCustomizer... definitionCustomizers) { 15 //將Bean配置類信息轉成容器中AnnotatedGenericBeanDefinition數據結構, AnnotatedGenericBeanDefinition繼承自BeanDefinition作用是定義一個bean的數據結構,下面的getMetadata可以獲取到該bean上的注解信息 16 AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass); 17 //@Conditional裝配條件判斷是否需要跳過注冊 18 if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) { 19 return; 20 } 21 //@param instanceSupplier a callback for creating an instance of the bean 22 //設置回調 23 abd.setInstanceSupplier(instanceSupplier); 24 //解析bean作用域(單例或者原型),如果有@Scope注解,則解析@Scope,沒有則默認為singleton 25 ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd); 26 //作用域寫回BeanDefinition數據結構, abd中缺損的情況下為空,將默認值singleton重新賦值到abd 27 abd.setScope(scopeMetadata.getScopeName()); 28 //生成bean配置類beanName 29 String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry)); 30 //通用注解解析到abd結構中,主要是處理Lazy, primary DependsOn, Role ,Description這五個注解 31 AnnotationConfigUtils.processCommonDefinitionAnnotations(abd); 32 //@param qualifiers specific qualifier annotations to consider, if any, in addition to qualifiers at the bean class level 33 // @Qualifier特殊限定符處理, 34 if (qualifiers != null) { 35 for (Class<? extends Annotation> qualifier : qualifiers) { 36 if (Primary.class == qualifier) { 37 // 如果配置@Primary注解,則設置當前Bean為自動裝配autowire時首選bean 38 abd.setPrimary(true); 39 } 40 else if (Lazy.class == qualifier) { 41 //設置當前bean為延遲加載 42 abd.setLazyInit(true); 43 } 44 else { 45 //其他注解,則添加到abd結構中 46 abd.addQualifier(new AutowireCandidateQualifier(qualifier)); 47 } 48 } 49 } 50 //自定義bean注冊,通常用在applicationContext創建后,手動向容器中一lambda表達式的方式注冊bean, 51 //比如:applicationContext.registerBean(UserService.class, () -> new UserService()); 52 for (BeanDefinitionCustomizer customizer : definitionCustomizers) { 53 //自定義bean添加到BeanDefinition 54 customizer.customize(abd); 55 } 56 //根據beanName和bean定義信息封裝一個beanhold,heanhold其實就是一個 beanname和BeanDefinition的映射 57 BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName); 58 //創建代理對象 59 definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry); 60 // BeanDefinitionReaderUtils.registerBeanDefinition 內部通過DefaultListableBeanFactory.registerBeanDefinition(String beanName, BeanDefinition beanDefinition)按名稱將bean定義信息注冊到容器中, 61 // 實際上DefaultListableBeanFactory內部維護一個Map<String, BeanDefinition>類型變量beanDefinitionMap,用於保存注bean定義信息(beanname 和 beandefine映射) 62 BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry); 63 }
register方法重點完成了bean配置類本身的解析和注冊,處理過程可以分為以下幾個步驟:
- 根據bean配置類,使用BeanDefinition解析Bean的定義信息,主要是一些注解信息
- Bean作用域的處理,默認缺少@Scope注解,解析成單例
- 借助AnnotationConfigUtils工具類解析通用注解
- 將bean定義信息已beanname,beandifine鍵值對的形式注冊到ioc容器中
3. refresh()刷新上下文
refresh方法在AbstractApplicationContext容器中實現,refresh()方法的作用加載或者刷新當前的配置信息,如果已經存在spring容器,則先銷毀之前的容器,重新創建spring容器,載入bean定義,完成容器初始化工作,debug進源碼可以看出AnnotationConfigApplicationContext容器是通過調用其父類AbstractApplicationContext的refresh()函數啟動整個IoC容器完成對Bean定義的載入。
AbstractApplicationContext.java中refresh方法的實現代碼如下:
1 public void refresh() throws BeansException, IllegalStateException { 2 synchronized (this.startupShutdownMonitor) { 3 //1.刷新上下文前的預處理 4 prepareRefresh(); 5 6 //2.獲取刷新后的內部Bean工廠 7 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); 8 9 //3.BeanFactory的預准備工作 10 prepareBeanFactory(beanFactory); 11 12 try { 13 // BeanFactory准備工作完成后,可以做一些后置處理工作, 14 // 4.空方法,用於在容器的子類中擴展 15 postProcessBeanFactory(beanFactory); 16 17 // 5. 執行BeanFactoryPostProcessor的方法,BeanFactory的后置處理器,在BeanFactory標准初始化之后執行的 18 invokeBeanFactoryPostProcessors(beanFactory); 19 20 // 6. 注冊BeanPostProcessor(Bean的后置處理器),用於攔截bean創建過程 21 registerBeanPostProcessors(beanFactory); 22 23 // 7. 初始化MessageSource組件(做國際化功能;消息綁定,消息解析) 24 initMessageSource(); 25 26 // 8. 初始化事件派發器 27 initApplicationEventMulticaster(); 28 29 // 9.空方法,可以用於子類實現在容器刷新時自定義邏輯 30 onRefresh(); 31 32 // 10. 注冊時間監聽器,將所有項目里面的ApplicationListener注冊到容器中來 33 registerListeners(); 34 35 // 11. 初始化所有剩下的單實例bean,單例bean在初始化容器時創建,原型bean在獲取時(getbean)時創建 36 finishBeanFactoryInitialization(beanFactory); 37 38 // 12. 完成BeanFactory的初始化創建工作,IOC容器就創建完成; 39 finishRefresh(); 40 } 41 42 catch (BeansException ex) { 43 if (logger.isWarnEnabled()) { 44 logger.warn("Exception encountered during context initialization - " + 45 "cancelling refresh attempt: " + ex); 46 } 47 48 // Destroy already created singletons to avoid dangling resources. 49 destroyBeans(); 50 51 // Reset 'active' flag. 52 cancelRefresh(ex); 53 54 // Propagate exception to caller. 55 throw ex; 56 } 57 58 finally { 59 // Reset common introspection caches in Spring's core, since we 60 // might not ever need metadata for singleton beans anymore... 61 resetCommonCaches(); 62 } 63 } 64 }
具體分析refresh中的函數邏輯:
1. 刷新上線文前的預處理 prepareRefresh():
AbstractApplicationContext. prepareRefresh ()方法:
1 protected void prepareRefresh() { 2 //設置容器啟動時間 3 this.startupDate = System.currentTimeMillis(); 4 //啟動標識 5 this.closed.set(false); 6 this.active.set(true); 7 8 if (logger.isInfoEnabled()) { 9 logger.info("Refreshing " + this); 10 } 11 12 //空方法,用於子容器自定義個性化的屬性設置方法 13 initPropertySources(); 14 //檢驗屬性的合法等 15 getEnvironment().validateRequiredProperties(); 16 17 //保存容器中的一些早期的事件 18 this.earlyApplicationEvents = new LinkedHashSet<>(); 19 }
2. 獲取刷新后的內部Bean工廠,obtainFreshBeanFactory方法為內部bean工廠重新生成id,並返回bean工廠
AbstractApplicationContext. obtainFreshBeanFactory()方法
1 protected ConfigurableListableBeanFactory obtainFreshBeanFactory() { 2 //為beanfactory生成唯一序列化id,beanfactory已經在GenericApplicationContext構造函數中初始化了,refreshBeanFactory的邏輯在AbstractApplicationContext的實現類GenericApplicationContext中 3 refreshBeanFactory(); 4 //獲取beanfactory 5 ConfigurableListableBeanFactory beanFactory = getBeanFactory(); 6 if (logger.isDebugEnabled()) { 7 logger.debug("Bean factory for " + getDisplayName() + ": " + beanFactory); 8 } 9 return beanFactory; 10 }
GenericApplicationContext.refreshBeanFactory()實現代碼
1 protected final void refreshBeanFactory() throws IllegalStateException { 2 if (!this.refreshed.compareAndSet(false, true)) { 3 throw new IllegalStateException( 4 "GenericApplicationContext does not support multiple refresh attempts: just call 'refresh' once"); 5 } 6 //生成一個序列化id 7 this.beanFactory.setSerializationId(getId()); 8 }
這里使用AbstractApplicationContext. refreshBeanFactory()在不同實現容器中有點區別,如果是以xml方式配置bean,會使用AbstractRefreshableApplicationContext容器中的實現,該容器中實現xml配置文件定位,並通過BeanDefinition載入和解析xml配置文件。
而如果是注解的方式,則並沒有解析項目包下的注解,而是通過在refresh()方法中執行ConfigurationClassPostProcessor后置處理器完成對bean的加載.
3.BeanFactory的預准備工作 prepareBeanFactory(beanFactory):
prepareBeanFactory主要完成beanFactory的一些屬性設置
1 protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) { 2 // Tell the internal bean factory to use the context's class loader etc. 3 beanFactory.setBeanClassLoader(getClassLoader()); //設置類加載器 4 beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader())); //bean表達式解析器 5 beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment())); 6 7 // Configure the bean factory with context callbacks. 8 beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this)); //添加一個BeanPostProcessor實現ApplicationContextAwareProcessor 9 //設置忽略的自動裝配接口,表示這些接口的實現類不允許通過接口自動注入 10 beanFactory.ignoreDependencyInterface(EnvironmentAware.class); 11 beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class); 12 beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class); 13 beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class); 14 beanFactory.ignoreDependencyInterface(MessageSourceAware.class); 15 beanFactory.ignoreDependencyInterface(ApplicationContextAware.class); 16 17 // BeanFactory interface not registered as resolvable type in a plain factory. 18 // MessageSource registered (and found for autowiring) as a bean. 19 //注冊可以自動裝配的組件,就是可以在任何組件中允許自動注入的組件 20 beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory); 21 beanFactory.registerResolvableDependency(ResourceLoader.class, this); 22 beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this); 23 beanFactory.registerResolvableDependency(ApplicationContext.class, this); 24 25 // Register early post-processor for detecting inner beans as ApplicationListeners. 26 beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this)); 27 28 //添加編譯時的AspectJ 29 if (beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) { 30 beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory)); 31 // Set a temporary ClassLoader for type matching. 32 beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader())); 33 } 34 35 // 給beanfactory容器中注冊組件ConfigurableEnvironment、systemProperties、systemEnvironment 36 if (!beanFactory.containsLocalBean(ENVIRONMENT_BEAN_NAME)) { 37 beanFactory.registerSingleton(ENVIRONMENT_BEAN_NAME, getEnvironment()); 38 } 39 if (!beanFactory.containsLocalBean(SYSTEM_PROPERTIES_BEAN_NAME)) { 40 beanFactory.registerSingleton(SYSTEM_PROPERTIES_BEAN_NAME, getEnvironment().getSystemProperties()); 41 } 42 if (!beanFactory.containsLocalBean(SYSTEM_ENVIRONMENT_BEAN_NAME)) { 43 beanFactory.registerSingleton(SYSTEM_ENVIRONMENT_BEAN_NAME, getEnvironment().getSystemEnvironment()); 44 } 45 }
5.執行bean工廠的后置處理器 invokeBeanFactoryPostProcessors(beanFactory)
IOC容器初始化過程中有三個重要的步驟,
1:資源定位,2:bean定義的載入,3:將bean名稱、bean定義以key-value形式注冊到容器,這三個步驟都將在此完成。
AbstractApplicationContext. invokeBeanFactoryPostProcessors方法實現:
1 protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) { 2 PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors()); 3 4 // Detect a LoadTimeWeaver and prepare for weaving, if found in the meantime 5 // (e.g. through an @Bean method registered by ConfigurationClassPostProcessor) 6 if (beanFactory.getTempClassLoader() == null && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) { 7 beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory)); 8 beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader())); 9 } 10 }
invokeBeanFactoryPostProcessors(beanFactory,getBeanFactoryPostProcessors())方法內部執行實現了BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor這兩個接口的Processor,先獲取所有BeanDefinitionRegistryPostProcessor的實現,按優先級執行(是否實現PriorityOrdered優先級接口,是否實現Ordered順序接口);再以相同的策略執行所有BeanFactoryPostProcessor的實現。
PostProcessorRegistrationDelegate. invokeBeanFactoryPostProcessors實現:
1 public static void invokeBeanFactoryPostProcessors( 2 ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) { 3 4 // Invoke BeanDefinitionRegistryPostProcessors first, if any. 5 Set<String> processedBeans = new HashSet<>(); 6 7 if (beanFactory instanceof BeanDefinitionRegistry) { 8 BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory; 9 List<BeanFactoryPostProcessor> regularPostProcessors = new ArrayList<>(); 10 List<BeanDefinitionRegistryPostProcessor> registryProcessors = new ArrayList<>(); 11 12 for (BeanFactoryPostProcessor postProcessor : beanFactoryPostProcessors) { 13 if (postProcessor instanceof BeanDefinitionRegistryPostProcessor) { 14 BeanDefinitionRegistryPostProcessor registryProcessor = 15 (BeanDefinitionRegistryPostProcessor) postProcessor; 16 registryProcessor.postProcessBeanDefinitionRegistry(registry); 17 registryProcessors.add(registryProcessor); 18 } 19 else { 20 regularPostProcessors.add(postProcessor); 21 } 22 } 23 24 // Do not initialize FactoryBeans here: We need to leave all regular beans 25 // uninitialized to let the bean factory post-processors apply to them! 26 // Separate between BeanDefinitionRegistryPostProcessors that implement 27 // PriorityOrdered, Ordered, and the rest. 28 List<BeanDefinitionRegistryPostProcessor> currentRegistryProcessors = new ArrayList<>(); 29 30 // First, invoke the BeanDefinitionRegistryPostProcessors that implement PriorityOrdered. 31 String[] postProcessorNames = 32 beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); 33 for (String ppName : postProcessorNames) { 34 if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { 35 currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); 36 processedBeans.add(ppName); 37 } 38 } 39 sortPostProcessors(currentRegistryProcessors, beanFactory); 40 registryProcessors.addAll(currentRegistryProcessors); 41 invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry); 42 currentRegistryProcessors.clear(); 43 44 // Next, invoke the BeanDefinitionRegistryPostProcessors that implement Ordered. 45 postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); 46 for (String ppName : postProcessorNames) { 47 if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) { 48 currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); 49 processedBeans.add(ppName); 50 } 51 } 52 sortPostProcessors(currentRegistryProcessors, beanFactory); 53 registryProcessors.addAll(currentRegistryProcessors); 54 invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry); 55 currentRegistryProcessors.clear(); 56 57 // Finally, invoke all other BeanDefinitionRegistryPostProcessors until no further ones appear. 58 boolean reiterate = true; 59 while (reiterate) { 60 reiterate = false; 61 postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); 62 for (String ppName : postProcessorNames) { 63 if (!processedBeans.contains(ppName)) { 64 currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); 65 processedBeans.add(ppName); 66 reiterate = true; 67 } 68 } 69 sortPostProcessors(currentRegistryProcessors, beanFactory); 70 registryProcessors.addAll(currentRegistryProcessors); 71 invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry); 72 currentRegistryProcessors.clear(); 73 } 74 75 // Now, invoke the postProcessBeanFactory callback of all processors handled so far. 76 invokeBeanFactoryPostProcessors(registryProcessors, beanFactory); 77 invokeBeanFactoryPostProcessors(regularPostProcessors, beanFactory); 78 } 79 80 else { 81 // Invoke factory processors registered with the context instance. 82 invokeBeanFactoryPostProcessors(beanFactoryPostProcessors, beanFactory); 83 } 84 85 // Do not initialize FactoryBeans here: We need to leave all regular beans 86 // uninitialized to let the bean factory post-processors apply to them! 87 String[] postProcessorNames = 88 beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false); 89 90 // Separate between BeanFactoryPostProcessors that implement PriorityOrdered, 91 // Ordered, and the rest. 92 List<BeanFactoryPostProcessor> priorityOrderedPostProcessors = new ArrayList<>(); 93 List<String> orderedPostProcessorNames = new ArrayList<>(); 94 List<String> nonOrderedPostProcessorNames = new ArrayList<>(); 95 for (String ppName : postProcessorNames) { 96 if (processedBeans.contains(ppName)) { 97 // skip - already processed in first phase above 98 } 99 else if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { 100 priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanFactoryPostProcessor.class)); 101 } 102 else if (beanFactory.isTypeMatch(ppName, Ordered.class)) { 103 orderedPostProcessorNames.add(ppName); 104 } 105 else { 106 nonOrderedPostProcessorNames.add(ppName); 107 } 108 } 109 110 // First, invoke the BeanFactoryPostProcessors that implement PriorityOrdered. 111 sortPostProcessors(priorityOrderedPostProcessors, beanFactory); 112 invokeBeanFactoryPostProcessors(priorityOrderedPostProcessors, beanFactory); 113 114 // Next, invoke the BeanFactoryPostProcessors that implement Ordered. 115 List<BeanFactoryPostProcessor> orderedPostProcessors = new ArrayList<>(); 116 for (String postProcessorName : orderedPostProcessorNames) { 117 orderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class)); 118 } 119 sortPostProcessors(orderedPostProcessors, beanFactory); 120 invokeBeanFactoryPostProcessors(orderedPostProcessors, beanFactory); 121 122 // Finally, invoke all other BeanFactoryPostProcessors. 123 List<BeanFactoryPostProcessor> nonOrderedPostProcessors = new ArrayList<>(); 124 for (String postProcessorName : nonOrderedPostProcessorNames) { 125 nonOrderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class)); 126 } 127 invokeBeanFactoryPostProcessors(nonOrderedPostProcessors, beanFactory); 128 129 // Clear cached merged bean definitions since the post-processors might have 130 // modified the original metadata, e.g. replacing placeholders in values... 131 beanFactory.clearMetadataCache(); 132 }
這里面在處理BeanDefinitionRegistryPostProcessors時有一個非常重要的過程,AnnotationConfigApplicationContext構造函數在初始化reader時為內部beanFactory容器初始化了一個id為org.springframework.context.annotation.internalConfigurationAnnotationProcessor的組件,這是一個ConfigurationClassPostProcessor組件,用來處理添加@Configuration注解的類,並將Bean定義注冊到BeanFactory中。
invokeBeanFactoryPostProcessors的實現過程是較復雜,一路debug可以發現最終在org.springframework.context.annotation.ComponentScanAnnotationParser#parse方法完成第一步:【資源文件的定位】,其實就是拿到配置類的所在的包名:

繼續查看doScan方法源碼:
protected Set<BeanDefinitionHolder> doScan(String... basePackages) { Assert.notEmpty(basePackages, "At least one base package must be specified"); Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>(); for (String basePackage : basePackages) {
//從指定的包中掃描需要裝載的類(bean) Set<BeanDefinition> candidates = findCandidateComponents(basePackage); for (BeanDefinition candidate : candidates) { ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate); candidate.setScope(scopeMetadata.getScopeName()); String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry); if (candidate instanceof AbstractBeanDefinition) { postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName); } if (candidate instanceof AnnotatedBeanDefinition) { AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate); } if (checkCandidate(beanName, candidate)) { BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName); definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry); beanDefinitions.add(definitionHolder);
//將掃描出的bean定義注冊到IOC容器的beanDefinitionMap中 registerBeanDefinition(definitionHolder, this.registry); } } } return beanDefinitions; }
可以看到這個方法完成了第二和第三步,findCandidateComponents(basePackages)根據指定的掃描路徑掃描並解析成beandefine, 后面通過registerBeanDefinition(definitionHolder,this.regsistry)將這些beandefine注冊到IOC容器,及添加到IOC容器的beanDefinitionMap中。
6.注冊BeanPostProcessor(Bean的后置處理器),用於攔截bean創建過程
注冊后置處理器的大致邏輯是:
1.獲取所有的 BeanPostProcessor
2.根據處理器實現的接口區分出4中類型:
a.實現PriorityOrdered接口的處理器
b.實現Ordered接口的處理器,
c.實現MergedBeanDefinitionPostProcessor接口的處理器,
d.普通后置處理器
3.按這個4中類型依次注冊到容器中
4.注冊一個特殊的后置處理器ApplicationListenerDetector,ApplicationListenerDetector本身也實現了MergedBeanDefinitionPostProcessor接口,有個問題,這個為什么沒有在上面c,d之間注冊,而是放到最后?
AbstractApplicationContext .registerBeanPostProcessors(beanFactory);實現邏輯:
1 protected void registerBeanPostProcessors(ConfigurableListableBeanFactory beanFactory) { 2 PostProcessorRegistrationDelegate.registerBeanPostProcessors(beanFactory, this); 3 }
1 public static void registerBeanPostProcessors( 2 ConfigurableListableBeanFactory beanFactory, AbstractApplicationContext applicationContext) { 3 4 String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false); 5 6 // Register BeanPostProcessorChecker that logs an info message when 7 // a bean is created during BeanPostProcessor instantiation, i.e. when 8 // a bean is not eligible for getting processed by all BeanPostProcessors. 9 int beanProcessorTargetCount = beanFactory.getBeanPostProcessorCount() + 1 + postProcessorNames.length; 10 beanFactory.addBeanPostProcessor(new BeanPostProcessorChecker(beanFactory, beanProcessorTargetCount)); 11 12 // Separate between BeanPostProcessors that implement PriorityOrdered, 13 // Ordered, and the rest. 14 //按優先級分類 15 List<BeanPostProcessor> priorityOrderedPostProcessors = new ArrayList<>(); 16 List<BeanPostProcessor> internalPostProcessors = new ArrayList<>(); 17 List<String> orderedPostProcessorNames = new ArrayList<>(); 18 List<String> nonOrderedPostProcessorNames = new ArrayList<>(); 19 for (String ppName : postProcessorNames) { 20 if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { 21 BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); 22 priorityOrderedPostProcessors.add(pp); 23 if (pp instanceof MergedBeanDefinitionPostProcessor) { 24 internalPostProcessors.add(pp); 25 } 26 } 27 else if (beanFactory.isTypeMatch(ppName, Ordered.class)) { 28 orderedPostProcessorNames.add(ppName); 29 } 30 else { 31 nonOrderedPostProcessorNames.add(ppName); 32 } 33 } 34 35 //先注冊實現PriorityOrdered接口的處理器,添加到beanfactory容器中beanFactory.addBeanPostProcessor(postProcessor); 36 sortPostProcessors(priorityOrderedPostProcessors, beanFactory); 37 registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors); 38 39 //注冊實現Ordered接口的處理器 40 List<BeanPostProcessor> orderedPostProcessors = new ArrayList<>(); 41 for (String ppName : orderedPostProcessorNames) { 42 BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); 43 orderedPostProcessors.add(pp); 44 if (pp instanceof MergedBeanDefinitionPostProcessor) { 45 internalPostProcessors.add(pp); 46 } 47 } 48 sortPostProcessors(orderedPostProcessors, beanFactory); 49 registerBeanPostProcessors(beanFactory, orderedPostProcessors); 50 51 // 注冊沒有實現Ordered或PriorityOrdered的處理器(nonOrderedPostProcessors) 52 List<BeanPostProcessor> nonOrderedPostProcessors = new ArrayList<>(); 53 for (String ppName : nonOrderedPostProcessorNames) { 54 BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class); 55 nonOrderedPostProcessors.add(pp); 56 if (pp instanceof MergedBeanDefinitionPostProcessor) { 57 internalPostProcessors.add(pp); 58 } 59 } 60 registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors); 61 62 // Finally, re-register all internal BeanPostProcessors. 63 //最后,重新注冊所有internal BeanPostProcessors(實現MergedBeanDefinitionPostProcessor接口的后置處理器 64 65 sortPostProcessors(internalPostProcessors, beanFactory); 66 registerBeanPostProcessors(beanFactory, internalPostProcessors); 67 68 //注冊ApplicationListenerDetector,用於Bean創建完時檢查是否是ApplicationListener 69 beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(applicationContext)); 70 }
7.初始化MessageSource組件(做國際化功能;消息綁定,消息解析)
AbstractApplicationContext .initMessageSource()方法實現代碼:
1 protected void initMessageSource() { 2 //獲取beanFactory 3 ConfigurableListableBeanFactory beanFactory = getBeanFactory(); 4 //判斷是否已經存在id為MESSAGE_SOURCE_BEAN_NAME的組件 5 if (beanFactory.containsLocalBean(MESSAGE_SOURCE_BEAN_NAME)) { 6 this.messageSource = beanFactory.getBean(MESSAGE_SOURCE_BEAN_NAME, MessageSource.class); 7 // Make MessageSource aware of parent MessageSource. 8 if (this.parent != null && this.messageSource instanceof HierarchicalMessageSource) { 9 HierarchicalMessageSource hms = (HierarchicalMessageSource) this.messageSource; 10 if (hms.getParentMessageSource() == null) { 11 // Only set parent context as parent MessageSource if no parent MessageSource 12 // registered already. 13 hms.setParentMessageSource(getInternalParentMessageSource()); 14 } 15 } 16 if (logger.isDebugEnabled()) { 17 logger.debug("Using MessageSource [" + this.messageSource + "]"); 18 } 19 } 20 else { 21 // Use empty MessageSource to be able to accept getMessage calls. 22 DelegatingMessageSource dms = new DelegatingMessageSource(); 23 dms.setParentMessageSource(getInternalParentMessageSource()); 24 this.messageSource = dms; 25 beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME, this.messageSource); 26 if (logger.isDebugEnabled()) { 27 logger.debug("Unable to locate MessageSource with name '" + MESSAGE_SOURCE_BEAN_NAME + 28 "': using default [" + this.messageSource + "]"); 29 } 30 } 31 }
8.初始化事件派發器
AbstractApplicationContext .initApplicationEventMulticaster()方法實現邏輯
1 protected void initApplicationEventMulticaster() { 2 //獲取BeanFactory 3 ConfigurableListableBeanFactory beanFactory = getBeanFactory(); 4 //如果有配置beanName為applicationEventMulticaster的事件派發器,則將其賦給容器中的applicationEventMulticaster對象 5 if (beanFactory.containsLocalBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME)) { 6 this.applicationEventMulticaster = 7 beanFactory.getBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, ApplicationEventMulticaster.class); 8 if (logger.isDebugEnabled()) { 9 logger.debug("Using ApplicationEventMulticaster [" + this.applicationEventMulticaster + "]"); 10 } 11 } 12 else { 13 //不存在,則創建一個SimpleApplicationEventMulticaster事件派發器,並注冊到beanfactory中 14 this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory); 15 beanFactory.registerSingleton(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, this.applicationEventMulticaster); 16 if (logger.isDebugEnabled()) { 17 logger.debug("Unable to locate ApplicationEventMulticaster with name '" + 18 APPLICATION_EVENT_MULTICASTER_BEAN_NAME + 19 "': using default [" + this.applicationEventMulticaster + "]"); 20 } 21 } 22 }
10. 注冊時間監聽器,將項目里面的ApplicationListener注冊到容器中來
registerListeners方法主要實現將事件監聽器添加到IOC容器中的事件派發器中,並在最后做了一個事件發布的邏輯(如果之前的步驟有產生事件,則將earlyApplicationEvents中保存的事件逐一發布)
AbstractApplicationContext .registerListeners()方法實現邏輯:
1 protected void registerListeners() { 2 // Register statically specified listeners first. 3 for (ApplicationListener<?> listener : getApplicationListeners()) { 4 getApplicationEventMulticaster().addApplicationListener(listener); 5 } 6 7 // Do not initialize FactoryBeans here: We need to leave all regular beans 8 // uninitialized to let post-processors apply to them! 9 String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false); 10 for (String listenerBeanName : listenerBeanNames) { 11 getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName); 12 } 13 14 // Publish early application events now that we finally have a multicaster... 15 Set<ApplicationEvent> earlyEventsToProcess = this.earlyApplicationEvents; 16 this.earlyApplicationEvents = null; 17 if (earlyEventsToProcess != null) { 18 for (ApplicationEvent earlyEvent : earlyEventsToProcess) { 19 getApplicationEventMulticaster().multicastEvent(earlyEvent); 20 } 21 } 22 }
11. 初始化所有剩下的單實例bean,單例bean在初始化容器時創建,原型bean在獲取時(getbean)時創建
AbstractApplicationContext.finishBeanFactoryInitialization(beanFactory);方法實現代碼:
1 protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) { 2 //組件轉換器相關 3 if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) && 4 beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) { 5 beanFactory.setConversionService( 6 beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)); 7 } 8 9 // Register a default embedded value resolver if no bean post-processor 10 // (such as a PropertyPlaceholderConfigurer bean) registered any before: 11 // at this point, primarily for resolution in annotation attribute values. 12 if (!beanFactory.hasEmbeddedValueResolver()) { 13 beanFactory.addEmbeddedValueResolver(strVal -> getEnvironment().resolvePlaceholders(strVal)); 14 } 15 16 //aspectj相關. 17 String[] weaverAwareNames = beanFactory.getBeanNamesForType(LoadTimeWeaverAware.class, false, false); 18 for (String weaverAwareName : weaverAwareNames) { 19 getBean(weaverAwareName); 20 } 21 22 // Stop using the temporary ClassLoader for type matching. 23 beanFactory.setTempClassLoader(null); 24 25 // Allow for caching all bean definition metadata, not expecting further changes. 26 beanFactory.freezeConfiguration(); 27 28 // 初始化后剩下的單實例bean 29 beanFactory.preInstantiateSingletons(); 30 }
DefaultListableBeanFactory. preInstantiateSingletons()方法實現邏輯:
1 public void preInstantiateSingletons() throws BeansException { 2 if (logger.isDebugEnabled()) { 3 logger.debug("Pre-instantiating singletons in " + this); 4 } 5 6 // Iterate over a copy to allow for init methods which in turn register new bean definitions. 7 // While this may not be part of the regular factory bootstrap, it does otherwise work fine. 8 //容器中所有bean名稱 9 List<String> beanNames = new ArrayList<>(this.beanDefinitionNames); 10 11 // Trigger initialization of all non-lazy singleton beans... 12 for (String beanName : beanNames) { 13 //獲取Bean的定義信息;RootBeanDefinition 14 RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName); 15 //非抽象,單例,非延遲加載 16 if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) { 17 //是否是FactoryBean 18 if (isFactoryBean(beanName)) { 19 // 通過"&beanName"獲取工廠Bean實例 20 Object bean = getBean(FACTORY_BEAN_PREFIX + beanName); 21 if (bean instanceof FactoryBean) { 22 final FactoryBean<?> factory = (FactoryBean<?>) bean; 23 boolean isEagerInit; 24 if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) { 25 isEagerInit = AccessController.doPrivileged((PrivilegedAction<Boolean>) 26 ((SmartFactoryBean<?>) factory)::isEagerInit, 27 getAccessControlContext()); 28 } 29 else { 30 isEagerInit = (factory instanceof SmartFactoryBean && 31 ((SmartFactoryBean<?>) factory).isEagerInit()); 32 } 33 if (isEagerInit) { 34 getBean(beanName); 35 } 36 } 37 } 38 else { 39 //不是FactoryBean,則利用getBean(beanName)實例化bean 40 getBean(beanName); 41 } 42 } 43 } 44 45 // Trigger post-initialization callback for all applicable beans... 46 for (String beanName : beanNames) { 47 Object singletonInstance = getSingleton(beanName); 48 if (singletonInstance instanceof SmartInitializingSingleton) { 49 final SmartInitializingSingleton smartSingleton = (SmartInitializingSingleton) singletonInstance; 50 if (System.getSecurityManager() != null) { 51 AccessController.doPrivileged((PrivilegedAction<Object>) () -> { 52 smartSingleton.afterSingletonsInstantiated(); 53 return null; 54 }, getAccessControlContext()); 55 } 56 else { 57 smartSingleton.afterSingletonsInstantiated(); 58 } 59 } 60 } 61 }
12. 完成BeanFactory的初始化創建工作,IOC容器就創建完成
AbstractApplicationContext.finishRefresh()實現邏輯:
1 protected void finishRefresh() { 2 // Clear context-level resource caches (such as ASM metadata from scanning). 3 clearResourceCaches(); 4 5 //初始化和生命周期有關的后置處理器LifecycleProcessor,默認DefaultLifecycleProcessor 6 initLifecycleProcessor(); 7 8 // 回調生命周期處理器 9 getLifecycleProcessor().onRefresh(); 10 11 //發布容器刷新完成事件:ContextRefreshedEvent 12 publishEvent(new ContextRefreshedEvent(this)); 13 14 LiveBeansView.registerApplicationContext(this); 15 }
以上基本分析了AnnotationConfigApplicationContext容器的初始化過程, Spring容器在啟動過程中,會先保存所有注冊進來的Bean的定義信息;Spring容器根據條件創建Bean實例,區分單例,還是原型,后置處理器等(后置處理器會在容器創建過程中通過getBean創建,並執行相應的邏輯);Spring容器在創建bean實例后,會使用多種后置處理器來增加bean的功能,比如處理自動注入,AOP,異步,這種后置處理器機制也豐富了bean的功能。
