Spring Boot 定義系統(tǒng)啟動任務(wù)的多種方式
在 Servlet/Jsp 項(xiàng)目中,如果涉及到系統(tǒng)任務(wù),例如在項(xiàng)目啟動階段要做一些數(shù)據(jù)初始化操作,這些操作有一個共同的特點(diǎn),只在項(xiàng)目啟動時進(jìn)行,以后都不再執(zhí)行,這里,容易想到web基礎(chǔ)中的三大組件( Servlet、Filter、Listener )之一 Listener ,這種情況下,一般定義一個 ServletContextListener,然后就可以監(jiān)聽到項(xiàng)目啟動和銷毀,進(jìn)而做出相應(yīng)的數(shù)據(jù)初始化和銷毀操作,例如下面這樣:
public class MyListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
//在這里做數(shù)據(jù)初始化操作
}
@Override
public void contextDestroyed(ServletContextEvent sce) {
//在這里做數(shù)據(jù)備份操作
}
}
當(dāng)然,這是基礎(chǔ) web 項(xiàng)目的解決方案,如果使用了 Spring Boot,那么我們可以使用更為簡便的方式。Spring Boot 中針對系統(tǒng)啟動任務(wù)提供了兩種解決方案,分別是 CommandLineRunner 和 ApplicationRunner,分別來看。
CommandLineRunner
使用 CommandLineRunner 時,首先自定義 MyCommandLineRunner1 并且實(shí)現(xiàn) CommandLineRunner 接口:
@Component
@Order(100)
public class MyCommandLineRunner1 implements CommandLineRunner {
@Override
public void run(String... args) throws Exception {
}
}
關(guān)于這段代碼,我做如下解釋:
1.首先通過 @Compoent 注解將 MyCommandLineRunner1 注冊為Spring容器中的一個 Bean。
2.添加 @Order注解,表示這個啟動任務(wù)的執(zhí)行優(yōu)先級,因?yàn)樵谝粋€項(xiàng)目中,啟動任務(wù)可能有多個,所以需要有一個排序。@Order 注解中,數(shù)字越小,優(yōu)先級越大,默認(rèn)情況下,優(yōu)先級的值為 Integer.MAX_VALUE,表示優(yōu)先級最低。
3.在 run 方法中,寫啟動任務(wù)的核心邏輯,當(dāng)項(xiàng)目啟動時,run方法會被自動執(zhí)行。
4.run 方法的參數(shù),來自于項(xiàng)目的啟動參數(shù),即項(xiàng)目入口類中,main方法的參數(shù)會被傳到這里。

此時啟動項(xiàng)目,run方法就會被執(zhí)行,至于參數(shù),可以通過兩種方式來傳遞,如果是在 IDEA 中,可以通過如下方式來配置參數(shù):
另一種方式,則是將項(xiàng)目打包,在命令行中啟動項(xiàng)目,然后啟動時在命令行傳入?yún)?shù),如下:
java -jar devtools-0.0.1-SNAPSHOT.jar 三國演義 西游記
注意,這里參數(shù)傳遞時沒有key,直接寫value即可,執(zhí)行結(jié)果如下:

ApplicationRunner
ApplicationRunner 和 CommandLineRunner 功能一致,用法也基本一致,唯一的區(qū)別主要體現(xiàn)在對參數(shù)的處理上,ApplicationRunner 可以接收更多類型的參數(shù)(ApplicationRunner 除了可以接收 CommandLineRunner 的參數(shù)之外,還可以接收 key/value形式的參數(shù))。
使用 ApplicationRunner ,自定義類實(shí)現(xiàn) ApplicationRunner 接口即可,組件注冊以及組件優(yōu)先級的配置都和 CommandLineRunner 一致,如下:
@Component
@Order(98)
public class MyApplicationRunner1 implements ApplicationRunner {
@Override
public void run(ApplicationArguments args) throws Exception {
List<String> nonOptionArgs = args.getNonOptionArgs();
System.out.println("MyApplicationRunner1>>>"+nonOptionArgs);
Set<String> optionNames = args.getOptionNames();
for (String key : optionNames) {
System.out.println("MyApplicationRunner1>>>"+key + ":" + args.getOptionValues(key));
}
String[] sourceArgs = args.getSourceArgs();
System.out.println("MyApplicationRunner1>>>"+Arrays.toString(sourceArgs));
}
}
當(dāng)項(xiàng)目啟動時,這里的 run 方法就會被自動執(zhí)行,關(guān)于 run 方法的參數(shù) ApplicationArguments ,我說如下幾點(diǎn):
1.args.getNonOptionArgs();可以用來獲取命令行中的無key參數(shù)(和CommandLineRunner一樣)。
2.args.getOptionNames();可以用來獲取所有key/value形式的參數(shù)的key。
3.args.getOptionValues(key));可以根據(jù)key獲取key/value 形式的參數(shù)的value。
4.args.getSourceArgs(); 則表示獲取命令行中的所有參數(shù)。
ApplicationRunner 定義完成后,傳啟動參數(shù)也是兩種方式,參數(shù)類型也有兩種,第一種和 CommandLineRunner 一致,第二種則是 --key=value 的形式,在 IDEA 中定義方式如下:

或者使用 如下啟動命令:
java -jar devtools-0.0.1-SNAPSHOT.jar 三國演義 西游記 --age=99
運(yùn)行結(jié)果如下:

總結(jié)
整體來說 ,這兩種的用法的差異不大 ,主要體現(xiàn)在對參數(shù)的處理上,小伙伴可以根據(jù)項(xiàng)目中的實(shí)際情況選擇合適的解決方案。
相關(guān)文章
簡單談?wù)刯ava中匿名內(nèi)部類構(gòu)造函數(shù)
這篇文章主要簡單給我們介紹了java中匿名內(nèi)部類構(gòu)造函數(shù),并附上了簡單的示例,有需要的小伙伴可以參考下。2015-11-11
SpringBoot整合Swagger接口文檔工具的流程步驟
我們在開發(fā)接口的時候,會將接口文檔給前端的開發(fā)者進(jìn)行對接,我們可以通過Postman或者Yapi等接口管理工具進(jìn)行編寫管理,實(shí)際開發(fā)中,接口的管理確實(shí)也應(yīng)該通過專業(yè)的工具管理,本文,我們就來談?wù)勗趺丛赟pringBoot整合Swagger接口文檔工具2023-08-08
Mybatis中Collection集合標(biāo)簽的使用詳解
這篇文章主要介紹了Mybatis中Collection集合標(biāo)簽的使用詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06
Java超詳細(xì)講解多線程中的Process與Thread
進(jìn)程process:在一定的環(huán)境下,把靜態(tài)的程序代碼運(yùn)行起來,通過使用不同的資源,來完成一定的任務(wù);線程thread:是程序中一個單一的順序控制流程。在單個進(jìn)程中同時運(yùn)行多個線程完成不同的工作,稱為多線程2022-05-05

