Spring核心方法refresh的使用解析
refresh 方法簡述
refresh()是 Spring 最核心的方法,沒有之一,上帝就是用這個方法創(chuàng)造了 Spring 的世界。
這是一個同步方法,用synchronized關(guān)鍵字來實現(xiàn)的。該方法包含以下12個核心方法(步驟)。
prepareRefresh() obtainFreshBeanFactory() prepareBeanFactory(beanFactory) postProcessBeanFactory(beanFactory) invokeBeanFactoryPostProcessors(beanFactory) registerBeanPostProcessors(beanFactory) initMessageSource() initApplicationEventMulticaster() onRefresh() registerListeners() finishBeanFactoryInitialization(beanFactory) finishRefresh()
refresh 12個步驟說明
一、prepareRefresh()
該方法用于容器刷新前的準備,包括設(shè)置上下文狀態(tài),獲取屬性,驗證必要的屬性等。
// 設(shè)置啟動時間 this.startupDate = System.currentTimeMillis(); // 1交給子類實現(xiàn),初始化屬性源 initPropertySources(); // 驗證所有標記為必須的屬性 getEnvironment().validateRequiredProperties();
開發(fā)者可以實現(xiàn)initPropertySources()方法,添加屬性或設(shè)置需要驗證的屬性,如:
System.out.println("開始校驗自定義配置"); getEnvironment().setRequiredProperties("my-name");
在執(zhí)行驗證屬性時,將會校驗是否有my-name配置。
二、obtainFreshBeanFactory()
該方法獲取新的beanFactory。該方法很簡單,刷新 BeanFactory 和獲取 getBeanFactory。
// 刷新 BeanFactory refreshBeanFactory(); // 獲取 getBeanFactory return getBeanFactory();
這兩個方法都是需要子類去實現(xiàn)的,如果是基于 xml 配置的方法啟動,在刷新階段將會做如下事情。
//創(chuàng)建 beanFactory DefaultListableBeanFactory beanFactory = createBeanFactory(); // 指定序列化 id beanFactory.setSerializationId(getId()); // 定制 beanFactory,設(shè)置相關(guān)屬性,包括是否允許覆蓋同名稱的不同定義的對象以及循環(huán)依賴及設(shè)置 @Autowired 和 @Qualifier 注解解析器 customizeBeanFactory(beanFactory); // 初始化 DocumentReader,并進行 XML 文件的讀取及解析,生成 BeanDefinition loadBeanDefinitions(beanFactory);
三、prepareBeanFactory(beanFactory)
該方法用于配置標準的 beanFactory ,設(shè)置 ClassLoader ,設(shè)置 SpEL 表達式解析器,添加忽略注入的接口,添加 bean ,添加 bean 后置處理器等。
// 設(shè)置 beanFactory 的類加載器 beanFactory.setBeanClassLoader(getClassLoader()); // 設(shè)置支持表達式解析器 beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader())); // 為 beanFactory 增加了一個默認的 propertyEditor ,這個主要是對 bean 的屬性等設(shè)置管理的一個工具 beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment())); // 添加部分BeanPostProcessor【ApplicationContextAwareProcessor】 beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this)); //設(shè)置忽略的自動裝配的接口EnvironmentAware、EmbeddedValueResolverAware、xxx; //這些接口的實現(xiàn)類不能通過類型來自動注入 beanFactory.ignoreDependencyInterface(EnvironmentAware.class); beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class); beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class); beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class); beanFactory.ignoreDependencyInterface(MessageSourceAware.class); beanFactory.ignoreDependencyInterface(ApplicationContextAware.class); // 注冊可以解析的自動裝配;我們能直接在任何組件中自動注入 beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory); beanFactory.registerResolvableDependency(ResourceLoader.class, this); beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this); beanFactory.registerResolvableDependency(ApplicationContext.class, this); // 增加對 AspectJ 的支持 if (beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) { ?? ?beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory)); ?? ?// Set a temporary ClassLoader for type matching. ?? ?beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader())); } // 環(huán)境信息ConfigurableEnvironment if (!beanFactory.containsLocalBean(ENVIRONMENT_BEAN_NAME)) { ?? ?beanFactory.registerSingleton(ENVIRONMENT_BEAN_NAME, getEnvironment()); } // 系統(tǒng)屬性,systemProperties【Map<String, Object>】 if (!beanFactory.containsLocalBean(SYSTEM_PROPERTIES_BEAN_NAME)) { ?? ?beanFactory.registerSingleton(SYSTEM_PROPERTIES_BEAN_NAME, getEnvironment().getSystemProperties()); } // 系統(tǒng)環(huán)境變量systemEnvironment【Map<String, Object>】 if (!beanFactory.containsLocalBean(SYSTEM_ENVIRONMENT_BEAN_NAME)) { ?? ?beanFactory.registerSingleton(SYSTEM_ENVIRONMENT_BEAN_NAME, getEnvironment().getSystemEnvironment()); }
四、postProcessBeanFactory(beanFactory)
模板方法,此時,所有的 beanDefinition 已經(jīng)加載,但是還沒有實例化允許在子類中對 beanFactory 進行擴展處理。
比如添加 ware 相關(guān)接口自動裝配設(shè)置,添加后置處理器等,是子類擴展 prepareBeanFactory(beanFactory) 的方法。
五、invokeBeanFactoryPostProcessors(beanFactory)
實例化并調(diào)用所有注冊的beanFactory后置處理器(實現(xiàn)接口BeanFactoryPostProcessor的bean,在beanFactory標準初始化之后執(zhí)行)。
// 執(zhí)行BeanFactoryPostProcessor的方法 PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());
在invokeBeanFactoryPostProcessors方法中,Spring 會先去找到所有的實現(xiàn)了BeanDefinitionRegistryPostProcessor的 BeanFactory 后置處理器,然后先執(zhí)行實現(xiàn)PriorityOrdered的,再執(zhí)行實現(xiàn)了Ordered的。
其中最著名的就是ConfigurationClassPostProcessor,用來掃描被 @Component 和 @Bean 標記的對象,并注冊其 BeanDefinition 元數(shù)據(jù)到 Spring 容器的 BeanDefinitionMap 中。然后再去獲取所有的 BeanFactory 后置處理器,去掉已經(jīng)執(zhí)行過的,也是根據(jù)排序依次執(zhí)行。
該方法結(jié)束后,Spring 上下文中已經(jīng)注冊并執(zhí)行了 BeanFactory 后置處理器,也將一部分 BeanDefinition 注冊了進來。
六、registerBeanPostProcessors(beanFactory)
顧名思義,該方法是注冊 Bean 的后置處理器。
// 把這件事委托給 PostProcessorRegistrationDelegate 來處理 PostProcessorRegistrationDelegate.registerBeanPostProcessors(beanFactory, this);
// 1. 獲取所有的 Bean 后置處理器的名字 String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false); // 2. 對 Bean 后置處理器分類 List<BeanPostProcessor> priorityOrderedPostProcessors = new ArrayList<>(); List<BeanPostProcessor> internalPostProcessors = new ArrayList<>(); List<String> orderedPostProcessorNames = new ArrayList<>(); List<String> nonOrderedPostProcessorNames = new ArrayList<>(); // 3. 注冊 Bean 后置處理器 registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors); registerBeanPostProcessors(beanFactory, orderedPostProcessors); registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors); registerBeanPostProcessors(beanFactory, internalPostProcessors); // 4. 注冊 ApplicationListener 探測器 beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(applicationContext));
獲取所有的 Bean 后置處理器的名字。
對 Bean 后置處理器分類。執(zhí)行完分類之后,所有的priorityOrderedPostProcessors都將成為一個 Bean 進入 Spring 容器中。
priorityOrderedPostProcessors
是所有實現(xiàn)了PriorityOrdered接口的后置處理器。internalPostProcessors
是所有內(nèi)置的后置處理器。orderedPostProcessorNames
實現(xiàn)了ordered接口的后置處理器。nonOrderedPostProcessorNames
沒有排序的后置處理器。
七、initMessageSource()
為上下文初始化 Message 源,即對不同語言的消息體進行國際化處理。
八、initApplicationEventMulticaster()
初始化事件廣播器,并放入 applicationEventMulticaster bean 中
九、onRefresh()
模板方法,在容器刷新的時候可以自定義邏輯,不同的Spring容器做不同的事情。
十、registerListeners()
注冊監(jiān)聽器。
// 1. 添加指定的監(jiān)聽器 for (ApplicationListener<?> listener : getApplicationListeners()) { ?? ?getApplicationEventMulticaster().addApplicationListener(listener); } // 2. 獲取所有實現(xiàn) ApplicationListener 的廣播器,并添加 String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false); for (String listenerBeanName : listenerBeanNames) { ?? ?getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName); }
十一、finishBeanFactoryInitialization(beanFactory)
實例化所有剩余的非懶加載單例,比如invokeBeanFactoryPostProcessors方法中根據(jù)各種注解解析出來的類,在這個時候都會被初始化。實例化的過程各種BeanPostProcessor開始起作用。
// 1. 凍結(jié)所有的 bean,已經(jīng)注冊的 bean 定義將不會被修改或任何進一步的處理 beanFactory.freezeConfiguration(); // 2. 實例化所有剩余的非懶加載的 bean beanFactory.preInstantiateSingletons();
// 3.獲取容器中所有 beanDefinition 的名稱 List<String> beanNames = new ArrayList<>(this.beanDefinitionNames); for (String beanName : beanNames) { ? ? // 根據(jù) beanName 獲取 BeanDefinition ?? ?RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName); ?? ?// 不是抽象的 && 是單例的 && 不是懶加載的 ?? ?if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) { ?? ??? ?if (isFactoryBean(beanName)) { ?? ??? ? ? ?// 如果是 FactoryBean,就先獲取 FactoryBean 實例 ?? ??? ??? ?Object bean = getBean(FACTORY_BEAN_PREFIX + beanName); ?? ??? ??? ?if (bean instanceof FactoryBean) { ?? ??? ??? ??? ?………… ?? ??? ??? ?} ?? ??? ?} ?? ??? ?else { ?? ??? ? ? ?// 如果不是 FactoryBean,就直接獲取 Bean ?? ??? ??? ?getBean(beanName); ?? ??? ?} ?? ?} }
十二、finishRefresh()
refresh做完之后需要做的其他事情。
// 清除上下文資源緩存(如掃描中的ASM元數(shù)據(jù)) scanning). clearResourceCaches(); // 初始化上下文的生命周期處理器,并刷新(找出Spring容器中實現(xiàn)了Lifecycle接口的bean并執(zhí)行start()方法) initLifecycleProcessor(); getLifecycleProcessor().onRefresh(); // 發(fā)布ContextRefreshedEvent事件告知對應(yīng)的ApplicationListener進行響應(yīng)的操作 publishEvent(new ContextRefreshedEvent(this));
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java把Map轉(zhuǎn)為對象的實現(xiàn)代碼
在項目開發(fā)中,經(jīng)常碰到map轉(zhuǎn)實體對象或者對象轉(zhuǎn)map的場景,工作中,很多時候我們可能比較喜歡使用第三方j(luò)ar包的API對他們進行轉(zhuǎn)化,但這里,我想通過反射的方式對他們做轉(zhuǎn)化,感興趣的同學(xué)跟著小編來看看吧2023-08-08Netty4之如何實現(xiàn)HTTP請求、響應(yīng)
這篇文章主要介紹了Netty4之如何實現(xiàn)HTTP請求、響應(yīng)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-04-04springboot如何為web層添加統(tǒng)一請求前綴
這篇文章主要介紹了springboot如何為web層添加統(tǒng)一請求前綴,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02IDEA配置Maven并版本統(tǒng)一管理的實現(xiàn)
本文主要介紹了IDEA配置Maven并版本統(tǒng)一管理的實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09SSH框架網(wǎng)上商城項目第19戰(zhàn)之訂單信息級聯(lián)入庫以及頁面緩存問題
這篇文章主要介紹了SSH框架網(wǎng)上商城項目第19戰(zhàn)之訂單信息級聯(lián)入庫以及頁面緩存問題,感興趣的小伙伴們可以參考一下2016-06-06SpringBoot?mybatis-plus使用json字段實戰(zhàn)指南
在現(xiàn)代應(yīng)用開發(fā)中經(jīng)常會使用JSON格式存儲和傳輸數(shù)據(jù),為了便捷地處理數(shù)據(jù)庫中的JSON字段,MyBatis-Plus提供了強大的JSON處理器,這篇文章主要給大家介紹了關(guān)于SpringBoot?mybatis-plus使用json字段的相關(guān)資料,需要的朋友可以參考下2024-01-01spring boot整合redis實現(xiàn)shiro的分布式session共享的方法
本篇文章主要介紹了spring boot整合redis實現(xiàn)shiro的分布式session共享的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-01-01