假設有這么一個需求,要求在項目啟動過程中,完成線程池的初始化,加密證書加載等功能,你會怎么做?如果沒想好答案,請接着往下看。今天介紹幾種在Spring Boot中進行資源初始化的方式,幫助大家解決和回答這個問題。
CommandLineRunner
定義初始化類 MyCommandLineRunner
實現 CommandLineRunner 接口,並實現它的 run() 方法,在該方法中編寫初始化邏輯
注冊成Bean,添加 @Component注解即可
示例代碼如下:
@Component public class MyCommandLineRunner implements CommandLineRunner { @Override public void run(String... args) throws Exception { System.out.println("...init resources by implements CommandLineRunner"); } }
實現了 CommandLineRunner 接口的 Component 會在所有 Spring Beans 初始化完成之后, 在 SpringApplication.run() 執行之前完成。下面通過加兩行打印來驗證我們的測試。
@SpringBootApplication public 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()"); } }
控制台打印結果如下。
... 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
- 定義初始化類
MyApplicationRunner
- 實現
ApplicationRunner
接口,並實現它的run()
方法,在該方法中編寫初始化邏輯 - 注冊成Bean,添加
@Component
注解即可 - 示例代碼如下:
@Component public class MyApplicationRunner implements ApplicationRunner { @Override public void run(ApplicationArguments applicationArguments) throws Exception { System.out.println("...init resources by implements ApplicationRunner"); } }
可以看到,通過實現 ApplicationRunner 接口,和通過實現 CommandLineRunner 接口都可以完成項目的初始化操作,實現相同的效果。兩者之間唯一的區別是 run() 方法中自帶的形參不相同,在 CommandLineRunner 中只是簡單的String... args形參,而 ApplicationRunner 則是包含了 ApplicationArguments 對象,可以幫助獲得更豐富的項目信息。
@Order
如果項目中既有實現了 ApplicationRunner 接口的初始化類,又有實現了 CommandLineRunner 接口的初始化類,那么會是哪一個先執行呢?測試告訴我們,答案是實現了 ApplicationRunner 接口的初始化類先執行,我想這點倒是不需要大家過分去關注為什么。但如果需要改變兩個初始化類之間的默認執行順序,那么使用 @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"); } }
最終,控制台中打印如下。通過控制台輸出我們發現, @Order
注解值越小,該初始化類也就越早執行。
。。。。。。(此處省略一堆打印信息) 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 public void postConstruct() { System.out.println("... PostConstruct"); }
啟動項目,控制台中最終打印如下。
... 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()
文末小結
綜上,使用 @PostConstruct
注解進行初始化操作的順序是最快的,前提是這些操作不能依賴於其它Bean的初始化完成。通過添加 @Order
注解,我們可以改變同層級之間不同Bean的加載順序。
————————————————
版權聲明:本文為CSDN博主「happyJared」的原創文章,遵循 CC 4.0 BY-SA 版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/qq_28804275/article/details/80891941