假設(shè)有這么一個(gè)需求,要求在項(xiàng)目啟動(dòng)過程中,完成線程池的初始化,加密證書加載等功能,你會(huì)怎么做?如果沒想好答案,請(qǐng)接著往下看。今天介紹幾種在Spring Boot中進(jìn)行資源初始化的方式,幫助大家解決和回答這個(gè)問題。
CommandLineRunner
示例代碼如下:
@Componentpublic class MyCommandLineRunner implements CommandLineRunner { @Override public void run(String... args) throws Exception { System.out.println("...init resources by implements CommandLineRunner"); } }
實(shí)現(xiàn)了 CommandLineRunner 接口的 Component 會(huì)在所有 Spring Beans 初始化完成之后, 在 SpringApplication.run() 執(zhí)行之前完成。下面通過加兩行打印來驗(yàn)證我們的測(cè)試。
@SpringBootApplicationpublic class DemoApplication { public static void main(String[] args) { System.out.println("... start SpringApplication.run()"); SpringApplication.run(DemoApplication.class, args); System.out.println("... end SpringApplication.run()"); } }
控制臺(tái)打印結(jié)果如下。
... start SpringApplication.run()
. ____ _ __ _ _
/// / ___'_ __ _ _(_)_ __ __ _ / / / /
( ( )/___ | '_ | '_| | '_ // _` | / / / /
/// ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_/__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v1.5.11.RELEASE)
。。。。。。(此處省略一堆打印信息)
2018-05-02 17:01:19.700 INFO 21236 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
...init resources by implements CommandLineRunner
2018-05-02 17:01:19.708 INFO 21236 --- [ main] cn.mariojd.demo.DemoApplication : Started DemoApplication in 2.282 seconds (JVM running for 3.125)
... end SpringApplication.run()
ApplicationRunner
示例代碼如下:
@Componentpublic class MyApplicationRunner implements ApplicationRunner { @Override public void run(ApplicationArguments applicationArguments) throws Exception { System.out.println("...init resources by implements ApplicationRunner"); } }
可以看到,通過實(shí)現(xiàn) ApplicationRunner 接口,和通過實(shí)現(xiàn) CommandLineRunner 接口都可以完成項(xiàng)目的初始化操作,實(shí)現(xiàn)相同的效果。兩者之間唯一的區(qū)別是 run() 方法中自帶的形參不相同,在 CommandLineRunner 中只是簡(jiǎn)單的String... args形參,而 ApplicationRunner 則是包含了 ApplicationArguments 對(duì)象,可以幫助獲得更豐富的項(xiàng)目信息。
ApplicationArguments
@Order
如果項(xiàng)目中既有實(shí)現(xiàn)了 ApplicationRunner 接口的初始化類,又有實(shí)現(xiàn)了 CommandLineRunner 接口的初始化類,那么會(huì)是哪一個(gè)先執(zhí)行呢?測(cè)試告訴我們,答案是實(shí)現(xiàn)了 ApplicationRunner 接口的初始化類先執(zhí)行,我想這點(diǎn)倒是不需要大家過分去關(guān)注為什么。但如果需要改變兩個(gè)初始化類之間的默認(rèn)執(zhí)行順序,那么使用 @Order 注解就可以幫助我們解決這個(gè)問題。
@Order@Component@Order(1)public class MyCommandLineRunner implements CommandLineRunner { @Override public void run(String... args) throws Exception { System.out.println("...init resources by implements CommandLineRunner"); }}
@Component@Order(2)public class MyApplicationRunner implements ApplicationRunner { @Override public void run(ApplicationArguments applicationArguments) throws Exception { System.out.println("...init resources by implements ApplicationRunner"); } }
最終,控制臺(tái)中打印如下。通過控制臺(tái)輸出我們發(fā)現(xiàn), @Order 注解值越小,該初始化類也就越早執(zhí)行。
。。。。。。(此處省略一堆打印信息)
2018-05-02 17:27:31.450 INFO 28304 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
...init resources by implements CommandLineRunner
...init resources by implements ApplicationRunner
2018-05-02 17:27:31.453 INFO 28304 --- [ main] cn.mariojd.demo.DemoApplication : Started DemoApplication in 2.086 seconds (JVM running for 2.977)
@PostConstruct
使用 @PostConstruct 注解同樣可以幫助我們完成資源的初始化操作,前提是這些初始化操作不需要依賴于其它Spring beans的初始化工作。
@PostConstruct
可以看到 @PostConstruct 注解是用在方法上的,寫一個(gè)方法測(cè)試一下吧。
@PostConstruct public void postConstruct() { System.out.println("... PostConstruct"); }
啟動(dòng)項(xiàng)目,控制臺(tái)中最終打印如下。
... start SpringApplication.run()
. ____ _ __ _ _
/// / ___'_ __ _ _(_)_ __ __ _ / / / /
( ( )/___ | '_ | '_| | '_ // _` | / / / /
/// ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_/__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v1.5.11.RELEASE)
。。。。。。(此處省略一堆打印信息)
... PostConstruct
。。。。。。(此處省略一堆打印信息)
2018-05-02 17:40:22.300 INFO 29796 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
...init resources by implements CommandLineRunner
...init resources by implements ApplicationRunner
2018-05-02 17:40:22.303 INFO 29796 --- [ main] cn.mariojd.demo.DemoApplication : Started DemoApplication in 2.387 seconds (JVM running for 3.267)
... end SpringApplication.run()
文末小結(jié)
綜上,使用 @PostConstruct 注解進(jìn)行初始化操作的順序是最快的,前提是這些操作不能依賴于其它Bean的初始化完成。通過添加 @Order 注解,我們可以改變同層級(jí)之間不同Bean的加載順序。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持VeVb武林網(wǎng)。
新聞熱點(diǎn)
疑難解答
圖片精選