摘要:重要的會在后面拎出來多帶帶詳解加鎖,防止在過程中,重啟或銷毀造成不必要的問題準備此上下文以進行刷新,設(shè)置其啟動日期和,活動標志以及執(zhí)行屬性源的任何初始化,校驗配置文件。以后所有的相關(guān)的操作其實是委托給這個實例來處理的。
1、先上測試代碼
public static void main(String[] args){ //配置文件來啟動一個 ApplicationContext ApplicationContext context = new ClassPathXmlApplicationContext("classpath:application.xml"); System.out.println("context 啟動成功"); //從context 中取出我們的 Bean,而不是用 new HelloServiceImpl() 這種方式 HelloService helloService = context.getBean(HelloService.class); // 這句將輸出: ok System.out.println(helloService.get(); } public interface HelloService { String get(); } public class HelloServiceImpl implements HelloService{ @Override public String get() { return "ok"; } }
在resource目錄下創(chuàng)建application.xml
2、我看一下new ClassPathXmlApplicationContext("classpath:application.xml")干了些什么
跟蹤可以發(fā)現(xiàn)ClassPathXmlApplicationContext繼承ApplicationContext
下面沿著代碼一步步看都干了些什么:
public ClassPathXmlApplicationContext(String configLocation) throws BeansException { this(new String[] {configLocation}, true, null); } public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent) throws BeansException { //獲取資源管理器 super(parent); //設(shè)置配置文件路徑,即classpath:application.xml,當然結(jié)果是個數(shù)組 setConfigLocations(configLocations); //這里是我們的重點,重新創(chuàng)建一個AbstractApplicationContext if (refresh) { refresh(); } } //AbstractApplicationContext中實現(xiàn)了super(parent) public AbstractApplicationContext() { this.resourcePatternResolver = getResourcePatternResolver(); }3、看一下refresh方法的執(zhí)行過程
一些不重要的就不把代碼貼出來了,只寫方法注釋。重要的會在后面拎出來多帶帶詳解
public void refresh() throws BeansException, IllegalStateException { //加鎖,防止在refresh過程中,重啟或銷毀造成不必要的問題 synchronized (this.startupShutdownMonitor) { // 準備此上下文以進行刷新,設(shè)置其啟動日期和,活動標志以及執(zhí)行屬性源的任何初始化,校驗 xml 配置文件。 prepareRefresh(); // 創(chuàng)建beanFactory,解析配置文件為一個個beanDefinition(類似map,key是beanname,value是bean屬性(總有的有,beanname,是否懶加載,作用范圍,依賴等等)),然后放到beanFactory中,詳細看步驟4 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // 設(shè)置 BeanFactory 的類加載器,添加幾個 BeanPostProcessor,手動注冊幾個特殊的 bean,詳情看步驟5 prepareBeanFactory(beanFactory); try { //允許上下文子類在初始化的時候?qū)eanFactory進行修改 postProcessBeanFactory(beanFactory); //調(diào)用所有beanFactoryPostProcessor對beanDefinite信息進行修改,詳情見步驟6 invokeBeanFactoryPostProcessors(beanFactory); // Register bean processors that intercept bean creation. registerBeanPostProcessors(beanFactory); // Initialize message source for this context. initMessageSource(); // Initialize event multicaster for this context. initApplicationEventMulticaster(); // Initialize other special beans in specific context subclasses. onRefresh(); // Check for listener beans and register them. registerListeners(); // Instantiate all remaining (non-lazy-init) singletons. finishBeanFactoryInitialization(beanFactory); // Last step: publish corresponding event. finishRefresh(); } catch (BeansException ex) { if (logger.isWarnEnabled()) { logger.warn("Exception encountered during context initialization - " + "cancelling refresh attempt: " + ex); } // Destroy already created singletons to avoid dangling resources. destroyBeans(); // Reset "active" flag. cancelRefresh(ex); // Propagate exception to caller. throw ex; } finally { // Reset common introspection caches in Spring"s core, since we // might not ever need metadata for singleton beans anymore... resetCommonCaches(); } } }4、ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory()執(zhí)行過程
protected ConfigurableListableBeanFactory obtainFreshBeanFactory() { //關(guān)閉舊的 BeanFactory (如果有),創(chuàng)建新的 BeanFactory,加載 Bean 定義、注冊 Bean 等 refreshBeanFactory(); //返回剛剛創(chuàng)建的beanFactory ConfigurableListableBeanFactory beanFactory = getBeanFactory(); if (logger.isDebugEnabled()) { logger.debug("Bean factory for " + getDisplayName() + ": " + beanFactory); } return beanFactory; } protected final void refreshBeanFactory() throws BeansException { if (hasBeanFactory()) {//是否已經(jīng)存在beanFactory destroyBeans();//銷毀所有bean closeBeanFactory();//關(guān)閉beanFactory } try { //創(chuàng)建DefaultListableBeanFactory,為什么用這個,可以參考beanFactory繼承結(jié)構(gòu) //ApplicationContext 繼承自 BeanFactory,但是它不應(yīng)該被理解為 BeanFactory 的實現(xiàn)類,而是說其內(nèi)部持有一個實例化的 BeanFactory(DefaultListableBeanFactory)。以后所有的 BeanFactory 相關(guān)的操作其實是委托給這個實例來處理的。 DefaultListableBeanFactory beanFactory = createBeanFactory(); //序列化id beanFactory.setSerializationId(getId()); //設(shè)置 BeanFactory 的兩個配置屬性:是否允許 Bean 覆蓋、是否允許循環(huán)引用 customizeBeanFactory(beanFactory); //解析配置文件的封裝成beandefinite,加載到beanFactory(解析過程就不多少了,想了解的同學可以看看) loadBeanDefinitions(beanFactory); synchronized (this.beanFactoryMonitor) { this.beanFactory = beanFactory; } } catch (IOException ex) { throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex); } } protected void customizeBeanFactory(DefaultListableBeanFactory beanFactory) { if (this.allowBeanDefinitionOverriding != null) { //設(shè)置是否允許bean被覆蓋 beanFactory.setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding); } if (this.allowCircularReferences != null) { //設(shè)置是否可以循環(huán)引用 beanFactory.setAllowCircularReferences(this.allowCircularReferences); } }
了解完后,我們回到步驟3繼續(xù)跟代碼
5、prepareBeanFactory(beanFactory)詳情protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) { // 設(shè)置類加載器 // return (this.classLoader != null ? this.classLoader : ClassUtils.getDefaultClassLoader()) //這里是當前ApplicationContext類的類加載器 beanFactory.setBeanClassLoader(getClassLoader()); //設(shè)置bean的表達式解析器 beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader())); //添加注冊編輯屬性 beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment())); // 添加一個beanPostProcessor,這個挺重要 //它會檢測該對象是否實現(xiàn)了xxxAware接口,并將相關(guān)的xxxAware實例注入給bean,如BeanNameAware等 //關(guān)于xxxAware的作用可以參考這個http://www.what21.com/article/view/b_java_1484728658282.html beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this)); // 下面幾行的意思就是,如果某個 bean 依賴于以下幾個接口的實現(xiàn)類,在自動裝配的時候忽略它們, // Spring 會通過其他方式來處理這些依賴。 beanFactory.ignoreDependencyInterface(EnvironmentAware.class); beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class); beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class); beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class); beanFactory.ignoreDependencyInterface(MessageSourceAware.class); beanFactory.ignoreDependencyInterface(ApplicationContextAware.class); // 下面是處理幾個特殊的bean注入相應(yīng)的值 // beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory); beanFactory.registerResolvableDependency(ResourceLoader.class, this); beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this); beanFactory.registerResolvableDependency(ApplicationContext.class, this); // 添加一個beanPostProcessor,在bean被實例化后,如果是ApplicationListener的子類,則加到listeners中 beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this)); // 如果bean是LoadTimeWeaver,則織入(在運行時織入,是和AOP相似,只是時機不一樣) if (beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) { beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory)); // 設(shè)置一個臨時的類加載器 beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader())); } // 手動注入一些系統(tǒng)環(huán)境、配置所需要的bean if (!beanFactory.containsLocalBean(ENVIRONMENT_BEAN_NAME)) { beanFactory.registerSingleton(ENVIRONMENT_BEAN_NAME, getEnvironment()); } if (!beanFactory.containsLocalBean(SYSTEM_PROPERTIES_BEAN_NAME)) { beanFactory.registerSingleton(SYSTEM_PROPERTIES_BEAN_NAME, getEnvironment().getSystemProperties()); } if (!beanFactory.containsLocalBean(SYSTEM_ENVIRONMENT_BEAN_NAME)) { beanFactory.registerSingleton(SYSTEM_ENVIRONMENT_BEAN_NAME, getEnvironment().getSystemEnvironment()); } } 我們回到步驟3繼續(xù)6、invokeBeanFactoryPostProcessors(beanFactory)
//實例化并調(diào)用所有已注冊的BeanFactoryPostProcessor bean protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) { //getBeanFactoryPostProcessors():獲取所有的beanFactoryPostProcessor PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors()); // Detect a LoadTimeWeaver and prepare for weaving, if found in the meantime // (e.g. through an @Bean method registered by ConfigurationClassPostProcessor) if (beanFactory.getTempClassLoader() == null && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) { beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory)); beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader())); } }
public static void invokeBeanFactoryPostProcessors( ConfigurableListableBeanFactory beanFactory, ListbeanFactoryPostProcessors) { // 記錄已經(jīng)遍歷的beanFactoryPostProcessor Set processedBeans = new HashSet (); if (beanFactory instanceof BeanDefinitionRegistry) { BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory; List regularPostProcessors = new LinkedList (); List registryPostProcessors = new LinkedList (); for (BeanFactoryPostProcessor postProcessor : beanFactoryPostProcessors) { //BeanDefinitionRegistryPostProcessor的子類放入registryPostProcessors中,其他放在regularPostProcessors中 if (postProcessor instanceof BeanDefinitionRegistryPostProcessor) { BeanDefinitionRegistryPostProcessor registryPostProcessor = (BeanDefinitionRegistryPostProcessor) postProcessor; registryPostProcessor.postProcessBeanDefinitionRegistry(registry); registryPostProcessors.add(registryPostProcessor); } else { regularPostProcessors.add(postProcessor); } } // Do not initialize FactoryBeans here: We need to leave all regular beans // uninitialized to let the bean factory post-processors apply to them! // Separate between BeanDefinitionRegistryPostProcessors that implement // PriorityOrdered, Ordered, and the rest. //獲取beanFactory中所有BeanDefinitionRegistryPostProcessor String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); // 調(diào)用實現(xiàn)PriorityOrdered的BeanDefinitionRegistryPostProcessors List priorityOrderedPostProcessors = new ArrayList (); for (String ppName : postProcessorNames) { if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); processedBeans.add(ppName); } } sortPostProcessors(beanFactory, priorityOrderedPostProcessors); registryPostProcessors.addAll(priorityOrderedPostProcessors); invokeBeanDefinitionRegistryPostProcessors(priorityOrderedPostProcessors, registry); // Next, invoke the BeanDefinitionRegistryPostProcessors that implement Ordered. postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); List orderedPostProcessors = new ArrayList (); for (String ppName : postProcessorNames) { if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) { orderedPostProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); processedBeans.add(ppName); } } sortPostProcessors(beanFactory, orderedPostProcessors); registryPostProcessors.addAll(orderedPostProcessors); invokeBeanDefinitionRegistryPostProcessors(orderedPostProcessors, registry); // Finally, invoke all other BeanDefinitionRegistryPostProcessors until no further ones appear. boolean reiterate = true; while (reiterate) { reiterate = false; postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); for (String ppName : postProcessorNames) { if (!processedBeans.contains(ppName)) { BeanDefinitionRegistryPostProcessor pp = beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class); registryPostProcessors.add(pp); processedBeans.add(ppName); pp.postProcessBeanDefinitionRegistry(registry); reiterate = true; } } } // Now, invoke the postProcessBeanFactory callback of all processors handled so far. invokeBeanFactoryPostProcessors(registryPostProcessors, beanFactory); invokeBeanFactoryPostProcessors(regularPostProcessors, beanFactory); } else { // Invoke factory processors registered with the context instance. invokeBeanFactoryPostProcessors(beanFactoryPostProcessors, beanFactory); } // Do not initialize FactoryBeans here: We need to leave all regular beans // uninitialized to let the bean factory post-processors apply to them! String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false); // Separate between BeanFactoryPostProcessors that implement PriorityOrdered, // Ordered, and the rest. List priorityOrderedPostProcessors = new ArrayList (); List orderedPostProcessorNames = new ArrayList (); List nonOrderedPostProcessorNames = new ArrayList (); for (String ppName : postProcessorNames) { if (processedBeans.contains(ppName)) { // skip - already processed in first phase above } else if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) { priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanFactoryPostProcessor.class)); } else if (beanFactory.isTypeMatch(ppName, Ordered.class)) { orderedPostProcessorNames.add(ppName); } else { nonOrderedPostProcessorNames.add(ppName); } } // First, invoke the BeanFactoryPostProcessors that implement PriorityOrdered. sortPostProcessors(beanFactory, priorityOrderedPostProcessors); invokeBeanFactoryPostProcessors(priorityOrderedPostProcessors, beanFactory); // Next, invoke the BeanFactoryPostProcessors that implement Ordered. List orderedPostProcessors = new ArrayList (); for (String postProcessorName : orderedPostProcessorNames) { orderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class)); } sortPostProcessors(beanFactory, orderedPostProcessors); invokeBeanFactoryPostProcessors(orderedPostProcessors, beanFactory); // Finally, invoke all other BeanFactoryPostProcessors. List nonOrderedPostProcessors = new ArrayList (); for (String postProcessorName : nonOrderedPostProcessorNames) { nonOrderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class)); } invokeBeanFactoryPostProcessors(nonOrderedPostProcessors, beanFactory); // Clear cached merged bean definitions since the post-processors might have // modified the original metadata, e.g. replacing placeholders in values... beanFactory.clearMetadataCache(); }
未完繼續(xù)
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/75964.html
摘要:本篇文章是源碼解析上的續(xù)集,上一篇文章介紹了使用的方式啟動,然后追蹤了容器的創(chuàng)建配置文件的解析的注冊等。前方超長篇幅預警。。。記錄依賴關(guān)系通過類型裝配。這也是作者第一次閱讀開源框架的源碼,如文章有錯誤之處還請您費心指出。 注意,看完這篇文章需要很長很長很長時間。。。 本篇文章是SpringIOC源碼解析(上)的續(xù)集,上一篇文章介紹了使用XML的方式啟動Spring,然后追蹤了BeanF...
摘要:不同與其它中間件框架,中有大量的業(yè)務(wù)代碼,它向我們展示了大神是如何寫業(yè)務(wù)代碼的依賴的層次結(jié)構(gòu),如何進行基礎(chǔ)包配置,以及工具類編寫,可以稱之為之最佳實踐。代碼參考視圖解析器,這里的配置指的是不檢查頭,而且默認請求為格式。 不同與其它中間件框架,Apollo中有大量的業(yè)務(wù)代碼,它向我們展示了大神是如何寫業(yè)務(wù)代碼的:maven依賴的層次結(jié)構(gòu),如何進行基礎(chǔ)包配置,以及工具類編寫,可以稱之為sp...
摘要:對于開發(fā)者來說,無疑是最常用也是最基礎(chǔ)的框架之一。概念上的東西還是要提一嘴的用容器來管理。和是容器的兩種表現(xiàn)形式。定義了簡單容器的基本功能。抽象出一個資源類來表示資源調(diào)用了忽略指定接口的自動裝配功能委托解析資源。 對于Java開發(fā)者來說,Spring無疑是最常用也是最基礎(chǔ)的框架之一。(此處省略1w字吹Spring)。相信很多同行跟我一樣,只是停留在會用的階段,比如用@Component...
摘要:即,的后置處理器,它的作用就是在的初始化方法前跟后進行攔截處理。如何注冊后置處理器我們暫不作分析,著重說一下,后置處理器是如何工作的。 BeanPostProcessor即,Bean的后置處理器,它的作用就是在Bean的初始化方法前跟后進行攔截處理。我們都知道,要想在Bean的初始化方法前后進行工作,那必須在Bean實例創(chuàng)建完成之后,init方法執(zhí)行之前,后置處理器就已經(jīng)在容器中了,所...
閱讀 2406·2021-10-09 09:44
閱讀 2140·2021-10-08 10:05
閱讀 3432·2021-07-26 23:38
閱讀 3008·2019-08-28 18:16
閱讀 820·2019-08-26 11:55
閱讀 1827·2019-08-23 18:29
閱讀 2042·2019-08-23 18:05
閱讀 1373·2019-08-23 17:02