亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring核心方法refresh的使用解析

 更新時(shí)間:2022年08月17日 15:17:43   作者:Jon?Kee  
這篇文章主要介紹了Spring核心方法refresh的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

refresh 方法簡(jiǎn)述

refresh()是 Spring 最核心的方法,沒有之一,上帝就是用這個(gè)方法創(chuàng)造了 Spring 的世界。

這是一個(gè)同步方法,用synchronized關(guān)鍵字來實(shí)現(xiàn)的。該方法包含以下12個(gè)核心方法(步驟)。

prepareRefresh()
obtainFreshBeanFactory()
prepareBeanFactory(beanFactory)
postProcessBeanFactory(beanFactory)
invokeBeanFactoryPostProcessors(beanFactory)
registerBeanPostProcessors(beanFactory)
initMessageSource()
initApplicationEventMulticaster()
onRefresh()
registerListeners()
finishBeanFactoryInitialization(beanFactory)
finishRefresh()

refresh 12個(gè)步驟說明

一、prepareRefresh()

該方法用于容器刷新前的準(zhǔn)備,包括設(shè)置上下文狀態(tài),獲取屬性,驗(yàn)證必要的屬性等。

// 設(shè)置啟動(dòng)時(shí)間
this.startupDate = System.currentTimeMillis();
// 1交給子類實(shí)現(xiàn),初始化屬性源
initPropertySources();
// 驗(yàn)證所有標(biāo)記為必須的屬性
getEnvironment().validateRequiredProperties();

開發(fā)者可以實(shí)現(xiàn)initPropertySources()方法,添加屬性或設(shè)置需要驗(yàn)證的屬性,如:

System.out.println("開始校驗(yàn)自定義配置");
getEnvironment().setRequiredProperties("my-name");

在執(zhí)行驗(yàn)證屬性時(shí),將會(huì)校驗(yàn)是否有my-name配置。

二、obtainFreshBeanFactory()

該方法獲取新的beanFactory。該方法很簡(jiǎn)單,刷新 BeanFactory 和獲取 getBeanFactory。

// 刷新 BeanFactory
refreshBeanFactory();
// 獲取 getBeanFactory
return getBeanFactory();

這兩個(gè)方法都是需要子類去實(shí)現(xiàn)的,如果是基于 xml 配置的方法啟動(dòng),在刷新階段將會(huì)做如下事情。

//創(chuàng)建 beanFactory
DefaultListableBeanFactory beanFactory = createBeanFactory();
// 指定序列化 id
beanFactory.setSerializationId(getId());
// 定制 beanFactory,設(shè)置相關(guān)屬性,包括是否允許覆蓋同名稱的不同定義的對(duì)象以及循環(huán)依賴及設(shè)置 @Autowired 和 @Qualifier 注解解析器
customizeBeanFactory(beanFactory);
// 初始化 DocumentReader,并進(jìn)行 XML 文件的讀取及解析,生成 BeanDefinition
loadBeanDefinitions(beanFactory);

三、prepareBeanFactory(beanFactory)

該方法用于配置標(biāo)準(zhǔn)的 beanFactory ,設(shè)置 ClassLoader ,設(shè)置 SpEL 表達(dá)式解析器,添加忽略注入的接口,添加 bean ,添加 bean 后置處理器等。

// 設(shè)置 beanFactory 的類加載器
beanFactory.setBeanClassLoader(getClassLoader());
// 設(shè)置支持表達(dá)式解析器
beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader()));
// 為 beanFactory 增加了一個(gè)默認(rèn)的 propertyEditor ,這個(gè)主要是對(duì) bean 的屬性等設(shè)置管理的一個(gè)工具
beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment()));
// 添加部分BeanPostProcessor【ApplicationContextAwareProcessor】
beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));
//設(shè)置忽略的自動(dòng)裝配的接口EnvironmentAware、EmbeddedValueResolverAware、xxx;
//這些接口的實(shí)現(xiàn)類不能通過類型來自動(dòng)注入
beanFactory.ignoreDependencyInterface(EnvironmentAware.class);
beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class);
beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class);
beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class);
beanFactory.ignoreDependencyInterface(MessageSourceAware.class);
beanFactory.ignoreDependencyInterface(ApplicationContextAware.class);
// 注冊(cè)可以解析的自動(dòng)裝配;我們能直接在任何組件中自動(dòng)注入
beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory);
beanFactory.registerResolvableDependency(ResourceLoader.class, this);
beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this);
beanFactory.registerResolvableDependency(ApplicationContext.class, this);
// 增加對(duì) 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)

模板方法,此時(shí),所有的 beanDefinition 已經(jīng)加載,但是還沒有實(shí)例化允許在子類中對(duì) beanFactory 進(jìn)行擴(kuò)展處理。

比如添加 ware 相關(guān)接口自動(dòng)裝配設(shè)置,添加后置處理器等,是子類擴(kuò)展 prepareBeanFactory(beanFactory) 的方法。

五、invokeBeanFactoryPostProcessors(beanFactory)

實(shí)例化并調(diào)用所有注冊(cè)的beanFactory后置處理器(實(shí)現(xiàn)接口BeanFactoryPostProcessor的bean,在beanFactory標(biāo)準(zhǔn)初始化之后執(zhí)行)。

// 執(zhí)行BeanFactoryPostProcessor的方法
PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());

在invokeBeanFactoryPostProcessors方法中,Spring 會(huì)先去找到所有的實(shí)現(xiàn)了BeanDefinitionRegistryPostProcessor的 BeanFactory 后置處理器,然后先執(zhí)行實(shí)現(xiàn)PriorityOrdered的,再執(zhí)行實(shí)現(xiàn)了Ordered的。

其中最著名的就是ConfigurationClassPostProcessor,用來掃描被 @Component 和 @Bean 標(biāo)記的對(duì)象,并注冊(cè)其 BeanDefinition 元數(shù)據(jù)到 Spring 容器的 BeanDefinitionMap 中。然后再去獲取所有的 BeanFactory 后置處理器,去掉已經(jīng)執(zhí)行過的,也是根據(jù)排序依次執(zhí)行。

該方法結(jié)束后,Spring 上下文中已經(jīng)注冊(cè)并執(zhí)行了 BeanFactory 后置處理器,也將一部分 BeanDefinition 注冊(cè)了進(jìn)來。

六、registerBeanPostProcessors(beanFactory)

顧名思義,該方法是注冊(cè) Bean 的后置處理器。

// 把這件事委托給 PostProcessorRegistrationDelegate 來處理
PostProcessorRegistrationDelegate.registerBeanPostProcessors(beanFactory, this);
// 1. 獲取所有的 Bean 后置處理器的名字
String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false);
// 2. 對(duì) Bean 后置處理器分類
List<BeanPostProcessor> priorityOrderedPostProcessors = new ArrayList<>();
List<BeanPostProcessor> internalPostProcessors = new ArrayList<>();
List<String> orderedPostProcessorNames = new ArrayList<>();
List<String> nonOrderedPostProcessorNames = new ArrayList<>();
// 3. 注冊(cè) Bean 后置處理器
registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors);
registerBeanPostProcessors(beanFactory, orderedPostProcessors);
registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors);
registerBeanPostProcessors(beanFactory, internalPostProcessors);
// 4. 注冊(cè) ApplicationListener 探測(cè)器
beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(applicationContext));

獲取所有的 Bean 后置處理器的名字。

對(duì) Bean 后置處理器分類。執(zhí)行完分類之后,所有的priorityOrderedPostProcessors都將成為一個(gè) Bean 進(jìn)入 Spring 容器中。

  • priorityOrderedPostProcessors是所有實(shí)現(xiàn)了PriorityOrdered接口的后置處理器。
  • internalPostProcessors是所有內(nèi)置的后置處理器。
  • orderedPostProcessorNames實(shí)現(xiàn)了ordered接口的后置處理器。
  • nonOrderedPostProcessorNames沒有排序的后置處理器。

七、initMessageSource()

為上下文初始化 Message 源,即對(duì)不同語言的消息體進(jìn)行國際化處理。

八、initApplicationEventMulticaster()

初始化事件廣播器,并放入 applicationEventMulticaster bean 中

九、onRefresh()

模板方法,在容器刷新的時(shí)候可以自定義邏輯,不同的Spring容器做不同的事情。

十、registerListeners()

注冊(cè)監(jiān)聽器。

// 1. 添加指定的監(jiān)聽器
for (ApplicationListener<?> listener : getApplicationListeners()) {
?? ?getApplicationEventMulticaster().addApplicationListener(listener);
}
// 2. 獲取所有實(shí)現(xiàn) ApplicationListener 的廣播器,并添加
String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);
for (String listenerBeanName : listenerBeanNames) {
?? ?getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);
}

十一、finishBeanFactoryInitialization(beanFactory)

實(shí)例化所有剩余的非懶加載單例,比如invokeBeanFactoryPostProcessors方法中根據(jù)各種注解解析出來的類,在這個(gè)時(shí)候都會(huì)被初始化。實(shí)例化的過程各種BeanPostProcessor開始起作用。

// 1. 凍結(jié)所有的 bean,已經(jīng)注冊(cè)的 bean 定義將不會(huì)被修改或任何進(jìn)一步的處理
beanFactory.freezeConfiguration();
// 2. 實(shí)例化所有剩余的非懶加載的 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 實(shí)例
?? ??? ??? ?Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);
?? ??? ??? ?if (bean instanceof FactoryBean) {
?? ??? ??? ??? ?…………
?? ??? ??? ?}
?? ??? ?}
?? ??? ?else {
?? ??? ? ? ?// 如果不是 FactoryBean,就直接獲取 Bean
?? ??? ??? ?getBean(beanName);
?? ??? ?}
?? ?}
}

十二、finishRefresh()

refresh做完之后需要做的其他事情。

// 清除上下文資源緩存(如掃描中的ASM元數(shù)據(jù)) scanning).
clearResourceCaches();
// 初始化上下文的生命周期處理器,并刷新(找出Spring容器中實(shí)現(xiàn)了Lifecycle接口的bean并執(zhí)行start()方法)
initLifecycleProcessor();
getLifecycleProcessor().onRefresh();
// 發(fā)布ContextRefreshedEvent事件告知對(duì)應(yīng)的ApplicationListener進(jìn)行響應(yīng)的操作
publishEvent(new ContextRefreshedEvent(this));

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。 

相關(guān)文章

  • Java把Map轉(zhuǎn)為對(duì)象的實(shí)現(xiàn)代碼

    Java把Map轉(zhuǎn)為對(duì)象的實(shí)現(xiàn)代碼

    在項(xiàng)目開發(fā)中,經(jīng)常碰到map轉(zhuǎn)實(shí)體對(duì)象或者對(duì)象轉(zhuǎn)map的場(chǎng)景,工作中,很多時(shí)候我們可能比較喜歡使用第三方j(luò)ar包的API對(duì)他們進(jìn)行轉(zhuǎn)化,但這里,我想通過反射的方式對(duì)他們做轉(zhuǎn)化,感興趣的同學(xué)跟著小編來看看吧
    2023-08-08
  • JDK9對(duì)String字符串的新一輪優(yōu)化

    JDK9對(duì)String字符串的新一輪優(yōu)化

    這篇文章主要介紹了JDK9對(duì)String字符串的新一輪優(yōu)化,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-03-03
  • Netty4之如何實(shí)現(xiàn)HTTP請(qǐng)求、響應(yīng)

    Netty4之如何實(shí)現(xiàn)HTTP請(qǐng)求、響應(yīng)

    這篇文章主要介紹了Netty4之如何實(shí)現(xiàn)HTTP請(qǐng)求、響應(yīng)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • springboot如何為web層添加統(tǒng)一請(qǐng)求前綴

    springboot如何為web層添加統(tǒng)一請(qǐng)求前綴

    這篇文章主要介紹了springboot如何為web層添加統(tǒng)一請(qǐng)求前綴,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • 詳解JVM系列之對(duì)象的鎖狀態(tài)和同步

    詳解JVM系列之對(duì)象的鎖狀態(tài)和同步

    鎖和同步是java多線程編程中非常常見的使用場(chǎng)景。為了鎖定多線程共享的對(duì)象,Java需要提供一定的機(jī)制來實(shí)現(xiàn)共享對(duì)象的鎖定。當(dāng)?shù)诙€(gè)線程進(jìn)入同一個(gè)區(qū)域的時(shí)候,必須等待第一個(gè)線程解鎖該對(duì)象。JVM是怎么做到的呢?快來一起看看吧。
    2021-06-06
  • IDEA配置Maven并版本統(tǒng)一管理的實(shí)現(xiàn)

    IDEA配置Maven并版本統(tǒng)一管理的實(shí)現(xiàn)

    本文主要介紹了IDEA配置Maven并版本統(tǒng)一管理的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • SSH框架網(wǎng)上商城項(xiàng)目第19戰(zhàn)之訂單信息級(jí)聯(lián)入庫以及頁面緩存問題

    SSH框架網(wǎng)上商城項(xiàng)目第19戰(zhàn)之訂單信息級(jí)聯(lián)入庫以及頁面緩存問題

    這篇文章主要介紹了SSH框架網(wǎng)上商城項(xiàng)目第19戰(zhàn)之訂單信息級(jí)聯(lián)入庫以及頁面緩存問題,感興趣的小伙伴們可以參考一下
    2016-06-06
  • java線程池詳解及代碼介紹

    java線程池詳解及代碼介紹

    這篇文章主要介紹了java中線程池的示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-08-08
  • SpringBoot?mybatis-plus使用json字段實(shí)戰(zhàn)指南

    SpringBoot?mybatis-plus使用json字段實(shí)戰(zhàn)指南

    在現(xiàn)代應(yīng)用開發(fā)中經(jīng)常會(huì)使用JSON格式存儲(chǔ)和傳輸數(shù)據(jù),為了便捷地處理數(shù)據(jù)庫中的JSON字段,MyBatis-Plus提供了強(qiáng)大的JSON處理器,這篇文章主要給大家介紹了關(guān)于SpringBoot?mybatis-plus使用json字段的相關(guān)資料,需要的朋友可以參考下
    2024-01-01
  • spring boot整合redis實(shí)現(xiàn)shiro的分布式session共享的方法

    spring boot整合redis實(shí)現(xiàn)shiro的分布式session共享的方法

    本篇文章主要介紹了spring boot整合redis實(shí)現(xiàn)shiro的分布式session共享的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-01-01

最新評(píng)論