Springboot中的@Order如何使用
在spring-boot 2.6.2下測(cè)試,@Order并不會(huì)影響bean的裝載順序,聲明了@Component的類,無(wú)論是構(gòu)造方法、@PostConstruct注解聲明的方法,還是實(shí)現(xiàn)的InitializingBean接口中的afterPropertiesSet()方法,如果beanClass位于同樣的目錄層級(jí),這些方法的調(diào)用只會(huì)受到className的順序影響:
@Component @Slf4j @Order(2) public class Bean1 implements InitializingBean { ? ? public Bean1() { ? ? ? ? log.info("construct bean1"); ? ? } ? ? @Override ? ? public void afterPropertiesSet() throws Exception { ? ? ? ? log.info("initialled bean1"); ? ? } ? ? @PostConstruct ? ? public void post() { ? ? ? ? log.info("post bean1"); ? ? } } @Component @Slf4j @Order(1) public class Bean2 implements InitializingBean { ? ? public Bean2() { ? ? ? ? log.info("construct bean2"); ? ? } ? ? @Override ? ? public void afterPropertiesSet() throws Exception { ? ? ? ? log.info("initialled bean2"); ? ? } ? ? @PostConstruct ? ? public void post() { ? ? ? ? log.info("post bean2"); ? ? } } /* 結(jié)果打印順序: construct bean1 post bean1 initialled bean1 construct bean2 post bean2 initialled bean2 */
觀察@Order的注解說(shuō)明,第一句寫著: @Order defines the sort order for an annotated component. 提到這個(gè)注解只是對(duì)component排序,那么哪里會(huì)收到這個(gè)排序數(shù)值的影響呢?
這里先改造一下代碼:
public interface IBean { ? ? void work(); } @Component @Slf4j @Order(2) public class Bean1 implements InitializingBean,CommandLineRunner,IBean { ? ? public Bean1() { ? ? ? ? log.info("construct bean1"); ? ? } ? ? @Override ? ? public void afterPropertiesSet() throws Exception { ? ? ? ? log.info("initialled bean1"); ? ? } ? ? @PostConstruct ? ? public void post() { ? ? ? ? log.info("post bean1"); ? ? } ? ? @Override ? ? public void run(String... args) throws Exception { ? ? ? ? log.info("running bean1"); ? ? } ? ? @Override ? ? public void work() { ? ? ? ? log.info("bean1 is working"); ? ? } } @Component @Slf4j @Order(1) public class Bean2 implements InitializingBean, CommandLineRunner, IBean { ? ? public Bean2() { ? ? ? ? log.info("construct bean2"); ? ? } ? ? @Override ? ? public void afterPropertiesSet() throws Exception { ? ? ? ? log.info("initialled bean2"); ? ? } ? ? @PostConstruct ? ? public void post() { ? ? ? ? log.info("post bean2"); ? ? } ? ? @Override ? ? public void run(String... args) throws Exception { ? ? ? ? log.info("running bean2"); ? ? } ? ? @Override ? ? public void work() { ? ? ? ? log.info("bean2 is working"); ? ? } } @Service @RequiredArgsConstructor public class TestService { ? ? private final List<IBean> beans; ? ? public void test(){ ? ? ? ? beans.forEach(IBean::work); ? ? } }
啟動(dòng)之后執(zhí)行TestService的test方法,得到如下順序的日志:
construct bean1
post bean1
initialled bean1
construct bean2
post bean2
initialled bean2
running bean2
running bean1
bean2 is working
bean1 is working
作一下說(shuō)明,@Order會(huì)影響依賴注入的順序,如果存在同樣類型的多個(gè)bean,且依賴聲明使用了List<BeanInterface>,會(huì)將所有bean實(shí)例按照Order聲明的順序放入一個(gè)ArrayList中注入,如果用的是Collection或者Set則無(wú)效,因?yàn)轭愋捅旧頍o(wú)序。
而CommandLineRunner聲明的run方法,會(huì)在bean被IOC容器裝配完成之后被調(diào)用,方法注釋簡(jiǎn)單明了的一句Callback used to run the bean可以理解為bean實(shí)例真正構(gòu)建完成之后的回調(diào)方法,而這個(gè)方法會(huì)受到@Order的順序影響,效果前面日志中已經(jīng)體現(xiàn),這里貼一下類注釋:
Interface used to indicate that a bean should run when it is contained within a SpringApplication. Multiple CommandLineRunner beans can be defined within the same application context and can be ordered using the Ordered interface or @Order annotation.
除了以上兩種用法,@Aspect聲明的切面類、繼承了OncePerRequestFilter的過(guò)濾器等,它們的作用順序也會(huì)受到Order的影響。
注意:如果@Order注解配置在了@Configuration修飾的配置類中的@Bean方法修飾的方法上時(shí),指定順序并不會(huì)生效
順便提一下另外一個(gè)注解:@Priority,以上@Order能起作用的地方,換成@Priority一樣會(huì)生效,但在一種情況下,它的作用和@Order大為不同:
同一個(gè)接口類型有多個(gè)不同的bean實(shí)現(xiàn)類時(shí),注入依賴時(shí)使用集合聲明不會(huì)報(bào)錯(cuò),但聲明為單體類型時(shí),如果各個(gè)Bean類使用了@Order聲明,就會(huì)報(bào)required a single bean, but x were found的錯(cuò)誤,這時(shí)有兩種方法可以解決問(wèn)題,一是在其中一個(gè)Bean類加上@Primary的注解聲明為首要類型,另外一個(gè)就是把Order改成Priority,優(yōu)先級(jí)最高的那個(gè)bean會(huì)被當(dāng)作primary來(lái)對(duì)待。
到此這篇關(guān)于Springboot中的@Order如何使用的文章就介紹到這了,更多相關(guān)Springboot @Order內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring中REST和RESTful的區(qū)別以及基本實(shí)現(xiàn)
本文主要介紹了spring中REST和RESTful的區(qū)別以及基本實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04IDEA2022版本創(chuàng)建maven?web項(xiàng)目的兩種方式詳解
創(chuàng)建maven?web項(xiàng)目有兩種方式,一種是使用骨架方式,一種是不使用骨架的方式,本文結(jié)合實(shí)例代碼給大家介紹了IDEA2022版本創(chuàng)建maven?web項(xiàng)目的兩種方式,需要的朋友可以參考下2023-02-02關(guān)于spring5的那些事:@Indexed 解密
這篇文章主要介紹了關(guān)于spring5的那些事:@Indexed 解密,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11Spring使用注解存儲(chǔ)Bean對(duì)象的方法詳解
在使用學(xué)習(xí)使用 Spring過(guò)程中,當(dāng)我們要實(shí)現(xiàn)一個(gè)功能的時(shí)候,先應(yīng)該考慮的是有沒有相應(yīng)的注解是實(shí)現(xiàn)對(duì)應(yīng)功能的,Spring 中很多功能的配置都是可以依靠注解實(shí)現(xiàn)的,而本篇中介紹的是使用注解來(lái)存儲(chǔ) Bean 對(duì)象2023-07-07詳解java如何處理各種批量數(shù)據(jù)入庫(kù)
這篇文章主要為大家詳細(xì)介紹了java如何使用BlockingQueue處理各種批量數(shù)據(jù)入庫(kù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-11-11Java Date類常用示例_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
在JDK1.0中,Date類是唯一的一個(gè)代表時(shí)間的類,但是由于Date類不便于實(shí)現(xiàn)國(guó)際化,所以從JDK1.1版本開始,推薦使用Calendar類進(jìn)行時(shí)間和日期處理。這里簡(jiǎn)單介紹一下Date類的使用,需要的朋友可以參考下2017-05-05Spring Cloud Feign實(shí)例講解學(xué)習(xí)
這篇文章主要介紹了Spring Cloud Feign實(shí)例講解學(xué)習(xí),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-02-02java后臺(tái)判斷客戶端是手機(jī)/PC并返回不同頁(yè)面的實(shí)例
下面小編就為大家分享一篇java后臺(tái)判斷客戶端是手機(jī)/PC并返回不同頁(yè)面的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01