SpringBoot Bean中如何指定初始化順序的若干姿勢(shì),很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。
我們提供的服務(wù)有:成都做網(wǎng)站、成都網(wǎng)站建設(shè)、微信公眾號(hào)開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、興山ssl等。為千余家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的興山網(wǎng)站制作公司
我們的測(cè)試項(xiàng)目和上一篇博文公用一個(gè)項(xiàng)目環(huán)境,當(dāng)然也可以建一個(gè)全新的測(cè)試項(xiàng)目,對(duì)應(yīng)的配置如下:(文末有源碼地址)
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.1.7</version> <relativePath/> <!-- lookup parent from update --> </parent> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> <spring-cloud.version>Finchley.RELEASE</spring-cloud.version> <java.version>1.8</java.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> <build> <pluginManagement> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </pluginManagement> </build> <repositories> <repository> <id>spring-milestones</id> <name>Spring Milestones</name> <url>https://repo.spring.io/milestone</url> <snapshots> <enabled>false</enabled> </snapshots> </repository> </repositories>
這種可以說是最簡(jiǎn)單也是最常見的使用姿勢(shì),但是在使用時(shí),需要注意循環(huán)依賴等問題
我們知道 bean 的注入方式之中,有一個(gè)就是通過構(gòu)造方法來注入,借助這種方式,我們可以解決有優(yōu)先級(jí)要求的 bean 之間的初始化順序
比如我們創(chuàng)建兩個(gè) Bean,要求 CDemo2 在 CDemo1 之前被初始化,那么我們的可用方式
@Component public class CDemo1 { private String name = "cdemo 1"; public CDemo1(CDemo2 cDemo2) { System.out.println(name); } } @Component public class CDemo2 { private String name = "cdemo 1"; public CDemo2() { System.out.println(name); } }
實(shí)測(cè)輸出結(jié)果如下,和我們預(yù)期一致
雖然這種方式比較直觀簡(jiǎn)單,但是有幾個(gè)限制
需要有注入關(guān)系,如 CDemo2 通過構(gòu)造方法注入到 CDemo1 中,如果需要指定兩個(gè)沒有注入關(guān)系的 bean 之間優(yōu)先級(jí),則不太合適(比如我希望某個(gè) bean 在所有其他的 Bean 初始化之前執(zhí)行)
循環(huán)依賴問題,如過上面的 CDemo2 的構(gòu)造方法有一個(gè) CDemo1 參數(shù),那么循環(huán)依賴產(chǎn)生,應(yīng)用無法啟動(dòng)
另外一個(gè)需要注意的點(diǎn)是,在構(gòu)造方法中,不應(yīng)有復(fù)雜耗時(shí)的邏輯,會(huì)拖慢應(yīng)用的啟動(dòng)時(shí)間
這是一個(gè)專用于解決 bean 的依賴問題,當(dāng)一個(gè) bean 需要在另一個(gè) bean 初始化之后再初始化時(shí),可以使用這個(gè)注解
使用方式也比較簡(jiǎn)單了,下面是一個(gè)簡(jiǎn)單的實(shí)例 case
@DependsOn("rightDemo2") @Component public class RightDemo1 { private String name = "right demo 1"; public RightDemo1() { System.out.println(name); } } @Component public class RightDemo2 { private String name = "right demo 2"; public RightDemo2() { System.out.println(name); } }
上面的注解放在 RightDemo1
上,表示RightDemo1
的初始化依賴于rightDemo2
這個(gè) bean
在使用這個(gè)注解的時(shí)候,有一點(diǎn)需要特別注意,它能控制 bean 的實(shí)例化順序,但是 bean 的初始化操作(如構(gòu)造 bean 實(shí)例之后,調(diào)用@PostConstruct
注解的初始化方法)順序則不能保證,比如我們下面的一個(gè)實(shí)例,可以說明這個(gè)問題
@DependsOn("rightDemo2") @Component public class RightDemo1 { private String name = "right demo 1"; @Autowired private RightDemo2 rightDemo2; public RightDemo1() { System.out.println(name); } @PostConstruct public void init() { System.out.println(name + " _init"); } } @Component public class RightDemo2 { private String name = "right demo 2"; @Autowired private RightDemo1 rightDemo1; public RightDemo2() { System.out.println(name); } @PostConstruct public void init() { System.out.println(name + " _init"); } }
注意上面的代碼,雖然說有循環(huán)依賴,但是通過@Autowired
注解方式注入的,所以不會(huì)導(dǎo)致應(yīng)用啟動(dòng)失敗,我們先看一下輸出結(jié)果
有意思的地方來了,我們通過@DependsOn
注解來確保在創(chuàng)建RightDemo1
之前,先得創(chuàng)建RightDemo2
;
所以從構(gòu)造方法的輸出可以知道,先實(shí)例 RightDemo2, 然后實(shí)例 RightDemo1;
然后從初始化方法的輸出可以知道,在上面這個(gè)場(chǎng)景中,雖然 RightDemo2 這個(gè) bean 創(chuàng)建了,但是它的初始化代碼在后面執(zhí)行
題外話: 有興趣的同學(xué)可以試一下把上面測(cè)試代碼中的
@Autowired
的依賴注入刪除,即兩個(gè) bean 沒有相互注入依賴,再執(zhí)行時(shí),會(huì)發(fā)現(xiàn)輸出順序又不一樣
最后再介紹一種非典型的使用方式,如非必要,請(qǐng)不要用這種方式來控制 bean 的加載順序
先創(chuàng)建兩個(gè)測(cè)試 bean
@Component public class HDemo1 { private String name = "h demo 1"; public HDemo1() { System.out.println(name); } } @Component public class HDemo2 { private String name = "h demo 2"; public HDemo2() { System.out.println(name); } }
我們希望 HDemo2 在 HDemo1 之前被加載,借助 BeanPostProcessor,我們可以按照下面的方式來實(shí)現(xiàn)
@Component public class DemoBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter implements BeanFactoryAware { private ConfigurableListableBeanFactory beanFactory; @Override public void setBeanFactory(BeanFactory beanFactory) { if (!(beanFactory instanceof ConfigurableListableBeanFactory)) { throw new IllegalArgumentException( "AutowiredAnnotationBeanPostProcessor requires a ConfigurableListableBeanFactory: " + beanFactory); } this.beanFactory = (ConfigurableListableBeanFactory) beanFactory; } @Override @Nullable public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) throws BeansException { // 在bean實(shí)例化之前做某些操作 if ("HDemo1".equals(beanName)) { HDemo2 demo2 = beanFactory.getBean(HDemo2.class); } return null; } }
請(qǐng)將目標(biāo)集中在postProcessBeforeInstantiation
,這個(gè)方法在某個(gè) bean 的實(shí)例化之前,會(huì)被調(diào)用,這就給了我們控制 bean 加載順序的機(jī)會(huì)
看到這種騷操作,是不是有點(diǎn)蠢蠢欲動(dòng),比如我有個(gè) bean,希望在應(yīng)用啟動(dòng)之后,其他的 bean 實(shí)例化之前就被加載,用這種方式是不是也可以實(shí)現(xiàn)呢?
下面是一個(gè)簡(jiǎn)單的實(shí)例 demo,重寫DemoBeanPostProcessor
的postProcessAfterInstantiation
方法,在 application 創(chuàng)建之后,就加載我們的 FDemo 這個(gè) bean
@Override public boolean postProcessAfterInstantiation(Object bean, String beanName) throws BeansException { if ("application".equals(beanName)) { beanFactory.getBean(FDemo.class); } return true; } @DependsOn("HDemo") @Component public class FDemo { private String name = "F demo"; public FDemo() { System.out.println(name); } } @Component public class HDemo { private String name = "H demo"; public HDemo() { System.out.println(name); } }
從下圖輸出可以看出,HDemo
, FDemo
的實(shí)例化順序放在了最前面了
在小結(jié)之前,先指出一下,一個(gè)完整的 bean 創(chuàng)建,在本文中區(qū)分了兩塊順序
實(shí)例化 (調(diào)用構(gòu)造方法)
初始化 (注入依賴屬性,調(diào)用@PostConstruct
方法)
主要介紹了三種方式來控制 bean 的加載順序,分別是
通過構(gòu)造方法依賴的方式,來控制有依賴關(guān)系的 bean 之間初始化順序,但是需要注意循環(huán)依賴的問題
@DependsOn
注解,來控制 bean 之間的實(shí)例順序,需要注意的是 bean 的初始化方法調(diào)用順序無法保證
BeanPostProcessor 方式,來手動(dòng)控制 bean 的加載順序
看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對(duì)創(chuàng)新互聯(lián)的支持。
文章題目:SpringBootBean中如何指定初始化順序的若干姿勢(shì)
URL地址:http://www.chinadenli.net/article48/ieodep.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供外貿(mào)網(wǎng)站建設(shè)、面包屑導(dǎo)航、App開發(fā)、關(guān)鍵詞優(yōu)化、企業(yè)網(wǎng)站制作、網(wǎng)站策劃
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)