【Java開發日記】說一說 SpringBoot 中 CommandLineRunner

目錄

1、CommandLineRunner

SpringBoot中CommandLineRunner的作用

簡單例子

多個類實現CommandLineRunner接口執行順序的保證

通過實現Ordered接口實現控制執行順序

通過@Order注解實現控制執行順序

@Order?作用

2、ApplicationRunner

3、傳遞參數

4、源碼跟蹤

run()方法

callRunners方法


1、CommandLineRunner

SpringBoot中CommandLineRunner的作用

平常開發中有可能需要實現在項目啟動后執行的功能,SpringBoot提供的一種簡單的實現方案就是添加一個model并實現CommandLineRunner接口,實現功能的代碼放在實現的run方法中。也就是項目一啟動之后,就立即需要執行的動作。只需要在項目里面簡單的配置,就可以實現這個功能。?

簡單例子
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;
@Component
public class MyStartupRunner implements CommandLineRunner {@Overridepublic void run(String... args) throws Exception {System.out.println("項目已經啟動");}
}

多個類實現CommandLineRunner接口執行順序的保證

通過實現Ordered接口實現控制執行順序
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.CommandLineRunner;
import org.springframework.core.Ordered;
import org.springframework.stereotype.Component;
/*** 優先級最高* 該類期望在springboot 啟動后第一順位執行* @since 12:57**/
@Slf4j
@Component
public class HighOrderCommandLineRunner implements CommandLineRunner, Ordered {@Overridepublic void run(String... args) throws Exception {for (String arg : args) {log.info("arg = " + arg);}log.info("i am highOrderRunner");}@Overridepublic int getOrder() {return Integer.MIN_VALUE+1;}
}
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.CommandLineRunner;
import org.springframework.core.Ordered;
import org.springframework.stereotype.Component;
/*** 優先級低于{@code HighOrderCommandLineRunner}* @since 12:59**/
@Slf4j
@Component
public class LowOrderCommandLineRunner implements CommandLineRunner, Ordered {@Overridepublic void run(String... args) throws Exception {log.info("i am lowOrderRunner");}@Overridepublic int getOrder() {return Integer.MIN_VALUE+1;}
}

啟動Spring Boot程序后,控制臺按照預定的順序打印出了結果:

2020-05-30 23:11:03.685  INFO 11976 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat started on port(s): 8080 (http) with context path ''
2020-05-30 23:11:03.701  INFO 11976 --- [           main] c.f.Application  : Started SpringBootApplication in 4.272 seconds (JVM running for 6.316)
2020-05-30 23:11:03.706  INFO 11976 --- [           main] c.f.HighOrderCommandLineRunner   : i am highOrderRunner
2020-05-30 23:11:03.706  INFO 11976 --- [           main] c.f.LowOrderCommandLineRunner   : i am lowOrderRunner

通過@Order注解實現控制執行順序

SpringBoot在項目啟動后會遍歷所有實現CommandLineRunner的實體類并執行run方法,如果需要按照一定的順序去執行,那么就需要在實體類上使用一個@Order注解(或者實現Order接口)來表明順序

import org.springframework.boot.CommandLineRunner;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
@Component
@Order(value=2)
public class MyStartupRunner1 implements CommandLineRunner {@Overridepublic void run(String... args) throws Exception {System.out.println("執行2");}
}
import org.springframework.boot.CommandLineRunner;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
@Component
@Order(value=1)
public class MyStartupRunner2 implements CommandLineRunner {@Overridepublic void run(String... args) throws Exception {System.out.println("執行1");}
}

控制臺顯示

執行1
執行2

根據控制臺結果可判斷,@Order?注解的執行優先級是按value值從小到大順序。

@Order?作用

項目啟動之后,要執行的動作是比較的多,那么到底先執行哪個,那么就可以利用這個注解限定優先級。 :::danger?Ordered接口并不能被?@Order注解所代替。

2、ApplicationRunner

在Spring Boot 1.3.0又引入了一個和CommandLineRunner功能一樣的接口ApplicationRunnerCommandLineRunner接收可變參數String... args,而ApplicationRunner?接收一個封裝好的對象參數ApplicationArguments。除此之外它們功能完全一樣,甚至連方法名都一樣。聲明一個ApplicationRunner并讓它優先級最低:

import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.core.Ordered;
import org.springframework.stereotype.Component;
import java.util.Arrays;
import java.util.List;
import java.util.Set;
/*** 優先級最低**/
@Slf4j
@Component
public class DefaultApplicationRunner implements ApplicationRunner, Ordered {@Overridepublic void run(ApplicationArguments args) throws Exception {log.info("i am applicationRunner");Set<String> optionNames = args.getOptionNames();log.info("optionNames = " + optionNames);String[] sourceArgs = args.getSourceArgs();log.info("sourceArgs = " + Arrays.toString(sourceArgs));List<String> nonOptionArgs = args.getNonOptionArgs();log.info("nonOptionArgs = " + nonOptionArgs);List<String> optionValues = args.getOptionValues("foo");log.info("optionValues = " + optionValues);}@Overridepublic int getOrder() {return Integer.MIN_VALUE+2;}
}

按照順序打印了三個類的執行結果:

2020-06-01 13:02:39.420  INFO 19032 --- [           main] c.f.MybatisResultmapApplication  : Started MybatisResultmapApplication in 1.801 seconds (JVM running for 2.266)
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.HighOrderCommandLineRunner   : i am highOrderRunner
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.LowOrderCommandLineRunner    : i am lowOrderRunner
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.DefaultApplicationRunner     : i am applicationRunner
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.DefaultApplicationRunner     : optionNames = []
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.DefaultApplicationRunner     : sourceArgs = []
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.DefaultApplicationRunner     : nonOptionArgs = []
2020-06-01 13:02:39.423  INFO 19032 --- [           main] c.f.DefaultApplicationRunner     : optionValues = null
 optionValues = null

Ordered接口并不能被?@Order注解所代替。

3、傳遞參數

Spring Boot應用啟動時是可以接受參數的,換句話說也就是Spring Bootmain方法是可以接受參數的。這些參數通過命令行?java -jar yourapp.jar?來傳遞。CommandLineRunner會原封不動照單全收這些接口,這些參數也可以封裝到ApplicationArguments對象中供ApplicationRunner調用。看一下ApplicationArguments的相關方法:

  • getSourceArgs()?被傳遞給應用程序的原始參數,返回這些參數的字符串數組。
  • getOptionNames()?獲取選項名稱的Set字符串集合。如?--spring.profiles.active=dev --debug?將返回["spring.profiles.active","debug"]?。
  • getOptionValues(String name)?通過名稱來獲取該名稱對應的選項值。如--foo=bar --foo=baz?將返回["bar","baz"]
  • containsOption(String name)?用來判斷是否包含某個選項的名稱。
  • getNonOptionArgs()?用來獲取所有的無選項參數。

可以通過下面的命令運行一個 Spring Boot應用 Jar

java -jar yourapp.jar --foo=bar --foo=baz --dev.name=fcant java fcantcn

或者在IDEA開發工具中打開Spring Boot應用main方法的配置項,進行命令行參數的配置,其他IDE工具同理。
運行Spring Boot應用,將會打印出:

2020-06-01 15:04:31.490  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : arg = --foo=bar
2020-06-01 15:04:31.490  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : arg = --foo=baz
2020-06-01 15:04:31.490  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : arg = --dev.name=fcant
2020-06-01 15:04:31.490  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : arg = java
2020-06-01 15:04:31.490  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : arg = fcantcn
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.HighOrderCommandLineRunner   : i am highOrderRunner
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.LowOrderCommandLineRunner    : i am lowOrderRunner
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.DefaultApplicationRunner     : i am applicationRunner
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.DefaultApplicationRunner     : optionNames = [dev.name, foo]
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.DefaultApplicationRunner     : sourceArgs = [--foo=bar, --foo=baz, --dev.name=fcant, java, fcantcn]
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.DefaultApplicationRunner     : nonOptionArgs = [java, fcantcn]
2020-06-01 15:04:31.491  INFO 13208 --- [           main] c.f.DefaultApplicationRunner     : optionValues = [bar, baz]

然后就可以根據實際需要動態地執行一些邏輯。?

4、源碼跟蹤

通過源碼理解一下底層實現。?

run()方法

跟進run方法后,一路F6直達以下方法

public ConfigurableApplicationContext run(String... args) {StopWatch stopWatch = new StopWatch();//設置線程啟動計時器stopWatch.start();ConfigurableApplicationContext context = null;Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList<>();//配置系統屬性:默認缺失外部顯示屏等允許啟動configureHeadlessProperty();//獲取并啟動事件監聽器,如果項目中沒有其他監聽器,則默認只有EventPublishingRunListenerSpringApplicationRunListeners listeners = getRunListeners(args);//將事件廣播給listenerslisteners.starting();try {//對于實現ApplicationRunner接口,用戶設置ApplicationArguments參數進行封裝ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);//配置運行環境:例如激活應用***.yml配置文件      ConfigurableEnvironment environment = prepareEnvironment(listeners,applicationArguments);configureIgnoreBeanInfo(environment);//加載配置的banner(gif,txt...),即控制臺圖樣Banner printedBanner = printBanner(environment);//創建上下文對象,并實例化context = createApplicationContext();exceptionReporters = getSpringFactoriesInstances(SpringBootExceptionReporter.class,new Class[] { ConfigurableApplicationContext.class }, context);//配置SPring容器      prepareContext(context, environment, listeners, applicationArguments,printedBanner);//刷新Spring上下文,創建bean過程中      refreshContext(context);//空方法,子類實現afterRefresh(context, applicationArguments);//停止計時器:計算線程啟動共用時間stopWatch.stop();if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);}//停止事件監聽器listeners.started(context);//開始加載資源callRunners(context, applicationArguments);}catch (Throwable ex) {handleRunFailure(context, listeners, exceptionReporters, ex);throw new IllegalStateException(ex);}listeners.running(context);return context;
}

主要是熟悉SpringBoot的CommandLineRunner接口實現原理。因此上面SpringBoot啟動過程方法不做過多介紹。直接進入CallRunners()方法內部。?

callRunners方法
private void callRunners(ApplicationContext context, ApplicationArguments args) {//將實現ApplicationRunner和CommandLineRunner接口的類,存儲到集合中List<Object> runners = new ArrayList<>();runners.addAll(context.getBeansOfType(ApplicationRunner.class).values());runners.addAll(context.getBeansOfType(CommandLineRunner.class).values());//按照加載先后順序排序AnnotationAwareOrderComparator.sort(runners);for (Object runner : new LinkedHashSet<>(runners)) {if (runner instanceof ApplicationRunner) {callRunner((ApplicationRunner) runner, args);}if (runner instanceof CommandLineRunner) {callRunner((CommandLineRunner) runner, args);}}
}
private void callRunner(CommandLineRunner runner, ApplicationArguments args) {try {//調用各個實現類中的邏輯實現(runner).run(args.getSourceArgs());}catch (Exception ex) {throw new IllegalStateException("Failed to execute CommandLineRunner", ex);}
}

到此結束,再跟進run()方法,就可以看到資源加載邏輯。

如果小假的內容對你有幫助,請點贊,評論,收藏。創作不易,大家的支持就是我堅持下去的動力!

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/86112.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/86112.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/86112.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

為什么React列表項需要key?(React key)(穩定的唯一標識key有助于React虛擬DOM優化重繪大型列表)

文章目錄 1. **幫助 React 識別列表項的變化**2. **性能優化**3. **避免組件狀態混亂**4. **為什么使用 rpid 作為 key**5. **不好的做法示例**6. **? 正確的做法** 在 React 中添加 key{item.rpid} 是非常重要的&#xff0c;主要有以下幾個原因&#xff1a; 1. 幫助 React 識…

算法筆記2

1.字符串拼接最好用StringBuilder&#xff0c;不用String 2.創建List<>類型的數組并創建內存 List arr[] new ArrayList[26]; Arrays.setAll(arr, i -> new ArrayList<>());

DeepSeek09-open-webui使用

Open WebUI 完全指南&#xff1a;從安裝到知識庫搭建與異常處理 最后更新&#xff1a;2025年6月7日 | 適用版本&#xff1a;Open WebUI v0.6.x 一、安裝部署 1.1 系統要求 **Python 3.12 **&#xff08;嚴格版本要求&#xff0c;更高版本3.13不兼容&#xff09;Node.js 20.x內…

前端面試五之vue2基礎

1.屬性綁定v-bind&#xff08;&#xff1a;&#xff09; v-bind 是 Vue 2 中用于動態綁定屬性的核心指令&#xff0c;它支持多種語法和用法&#xff0c;能夠靈活地綁定 DOM 屬性、組件 prop&#xff0c;甚至動態屬性名。通過 v-bind&#xff0c;可以實現數據與視圖之間的高效同…

408第一季 - 數據結構 - 棧與隊列

棧 閑聊 棧是一個線性表 棧的特點是后進先出 然后是一個公式 比如123要入棧&#xff0c;一共有5種排列組合的出棧 棧的數組實現 這里有兩種情況&#xff0c;&#xff0c;一個是有下標為-1的&#xff0c;一個沒有 代碼不用看&#xff0c;真題不會考 棧的鏈式存儲結構 L ->…

Linux(14)——庫的制作與原理

庫制作與原理技術文章大綱 庫的基本概念與分類 定義&#xff1a;庫&#xff08;Library&#xff09;在編程中的核心作用與意義分類&#xff1a;靜態庫&#xff08;Static Library&#xff09;、動態庫&#xff08;Dynamic Library&#xff09;的差異與應用場景常見示例&#…

2025政務服務便民熱線創新發展會議順利召開,張晨博士受邀分享

5月28日&#xff0c;由新華社中國經濟信息社、新華社廣東分社聯合主辦的2025政務服務便民熱線創新發展暨“人工智能熱線”會議在廣州舉行。會議圍繞“人工智能與新質熱線”主題&#xff0c;邀請全國的12345政務服務便民熱線主管部門負責人、省市熱線負責人和專家學者&#xff0…

AI驅動的B端頁面革命:智能布局、數據洞察的底層技術解析

摘要 ** 當企業 B 端系統的頁面還在依賴設計師反復調整布局&#xff0c;靠人工熬夜分析數據時&#xff0c;競爭對手已借助 AI 實現頁面的自動優化與智能決策。為何有的 B 端系統界面混亂&#xff0c;操作繁瑣&#xff0c;而 AI 賦能的頁面卻能精準適配用戶需求&#xff0c;秒…

大故障:阿里云核心域名爆炸了

大故障&#xff1a;阿里云核心域名被拖走了 今天早上許多群里出現網站故障的討論&#xff0c;比如 cnblogs 全國訪問一片紅&#xff0c;一看原來是阿里云又出故障了。 今天早上許多群里出現網站故障的討論&#xff0c;比如 cnblogs 全國訪問一片紅&#xff0c;一看原來是阿里云…

第1講、包管理和環境管理工具Conda 全面介紹

1. Conda 的背景與核心概念 1.1 什么是 Conda&#xff1f; Conda 是一個開源的、跨平臺的、語言無關的包管理和環境管理系統。它最初由 Anaconda 公司開發&#xff0c;旨在解決 Python 數據科學家面臨的包管理挑戰&#xff0c;但現在已經發展成為一個適用于多種編程語言的通用…

第4天:RNN應用(心臟病預測)

&#x1f368; 本文為&#x1f517;365天深度學習訓練營 中的學習記錄博客&#x1f356; 原作者&#xff1a;K同學啊 目標 具體實現 &#xff08;一&#xff09;環境 語言環境&#xff1a;Python 3.10 編 譯 器: PyCharm 框 架: Pytorch &#xff08;二&#xff09;具體步驟…

STM32學習筆記:外部中斷(EXTI)原理與應用詳解

前言 在嵌入式系統開發中&#xff0c;中斷機制是提高系統實時性和效率的重要手段。相比傳統的51單片機&#xff0c;STM32微控制器提供了更為豐富和靈活的外部中斷資源。本文將全面介紹STM32的外部中斷(EXTI)功能&#xff0c;包括其工作原理、配置方法和實際應用技巧。 一、外…

嵌入式知識篇---Zigbee串口

在 Python 中&#xff0c;serial和pyserial是經常被提及的兩個庫&#xff0c;它們在串口通信方面有著緊密的聯系&#xff0c;但又存在一些差異。下面將對它們進行詳細介紹&#xff0c;并給出各自的適用場景。 1. 基本概念 pyserial&#xff1a;它是 Python 里專門用于串口通信…

vue中的派發事件與廣播事件,及廣播事件應用于哪些場景和一個表單驗證例子

在 Vue 2.X 中&#xff0c;$dispatch 和 $broadcast 方法已經被廢棄。官方認為基于組件樹結構的事件流方式難以理解&#xff0c;并且在組件結構擴展時容易變得脆弱。因此&#xff0c;Vue 2.X 推薦使用其他方式來實現組件間的通信&#xff0c;例如通過 $emit 和 $on 方法&#x…

阿里云事件總線 EventBridge 正式商業化,構建智能化時代的企業級云上事件樞紐

作者&#xff1a;肯夢、稚柳 產品演進歷程&#xff1a;在技術浪潮中的成長之路 早在 2018 年&#xff0c;Gartner 評估報告便將事件驅動模型&#xff08;Event-Driven Model&#xff09;列為十大戰略技術趨勢之一&#xff0c;指出事件驅動架構&#xff08;EDA&#xff0c;Eve…

《前端面試題:BFC(塊級格式化上下文)》

前端BFC完全指南&#xff1a;布局魔法與面試必備 &#x1f38b; 端午安康&#xff01; 各位前端探險家&#xff0c;端午節快樂&#xff01;&#x1f96e; 愿你的代碼如龍舟競渡般乘風破浪&#xff0c;樣式如香糯粽子般完美包裹&#xff01;今天我們來解鎖CSS中的布局魔法——B…

dvwa10——XSS(DOM)

XSS攻擊&#xff1a; DOM型XSS 只在瀏覽器前端攻擊觸發&#xff1a;修改url片段代碼不存儲 反射型XSS 經過服務器攻擊觸發&#xff1a;可能通過提交惡意表單&#xff0c;連接觸發代碼不存儲 存儲型XSS 經由服務器攻擊觸發&#xff1a;可能通過提交惡意表單&#xff0c;連…

跨平臺資源下載工具:res-downloader 的使用體驗

一款基于 Go Wails 的跨平臺資源下載工具&#xff0c;簡潔易用&#xff0c;支持多種資源嗅探與下載。res-downloader 一款開源免費的下載軟件(開源無毒、放心使用)&#xff01;支持Win10、Win11、Mac系統.支持視頻、音頻、圖片、m3u8等網絡資源下載.支持視頻號、小程序、抖音、…

AOSP CachedAppOptimizer中的凍結和內存壓縮功能

AOSP CachedAppOptimizer&#xff1a;應用進程長期處于 Cached 狀態的內存壓縮和凍結優化管控 凍結和內存壓縮兩個功能獨立觸發&#xff0c;可以單獨觸發也可以組合觸發&#xff0c;默認順序&#xff1a;先壓縮&#xff0c;后凍結 public class OomAdjuster { protected b…

相機--相機成像原理和基礎概念

教程 成像原理 基礎概念 焦距&#xff08;物理焦距&#xff09; 鏡頭的光學中心到感光元件之間的距離&#xff0c;用f表示&#xff0c;單位&#xff1a;mm&#xff1b;。 像素焦距 相機內參矩陣中的 fx? 和 fy? 是將物理焦距轉換到像素坐標系的產物&#xff0c;可能不同。…