日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 嵌入式 > wenzi嵌入式軟件
[導(dǎo)讀]筆者能力有限,如果文中有錯(cuò)誤的地方,歡迎各位朋友給我及時(shí)地指出來,我將不甚感激,謝謝~ 概念 引用維基百科上的關(guān)于回調(diào)函數(shù)的概念: 在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡(jiǎn)稱回調(diào)(Callback 即call then back 被主函數(shù)調(diào)用運(yùn)算后會(huì)返回主函數(shù)),是指通過函

筆者能力有限,如果文中有錯(cuò)誤的地方,歡迎各位朋友給我及時(shí)地指出來,我將不甚感激,謝謝~

概念

引用維基百科上的關(guān)于回調(diào)函數(shù)的概念:

在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡(jiǎn)稱回調(diào)(Callback 即call then back 被主函數(shù)調(diào)用運(yùn)算后會(huì)返回主函數(shù)),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計(jì)允許了底層代碼調(diào)用在高層定義的子程序。

打一個(gè)簡(jiǎn)單的例子就是說,如果我們?cè)谝粋€(gè) RTOS 的基礎(chǔ)上去編寫應(yīng)用程序,編寫應(yīng)用程序的這一層就是應(yīng)用層,也可以說是高層,那 RTOS 內(nèi)核所處的就是內(nèi)核層,也可以說是底層。在編寫應(yīng)用程序的時(shí)候,我們可以函數(shù)調(diào)用的形式來在高層調(diào)用底層的函數(shù)來實(shí)現(xiàn)相關(guān)的功能,但是底層的程序在使用過程中,一般是不進(jìn)行改動(dòng)的,也就無法通過普通函數(shù)調(diào)用的方法去調(diào)用在高層定義的函數(shù),而回調(diào)函數(shù)則能解決這一問題,使得底層代碼調(diào)用在高層定義的子程序,下面通過一個(gè)圖簡(jiǎn)單說明這個(gè)問題:

回調(diào)函數(shù)的實(shí)現(xiàn)

對(duì)于回調(diào)函數(shù)一種比較簡(jiǎn)單的理解也就是將一個(gè)函數(shù)指針以參數(shù)的形式傳遞給另一個(gè)函數(shù),在這里不對(duì)函數(shù)指針的概念進(jìn)行展開講解,筆者在《C 語言跳轉(zhuǎn)表的實(shí)現(xiàn)及在嵌入式設(shè)備中的應(yīng)用》中簡(jiǎn)單地描述了函數(shù)指針的概念。在大多數(shù)情況下,回調(diào)函數(shù)將包括以下三個(gè)部分:

  • 定義回調(diào)函數(shù)

  • 注冊(cè)回調(diào)函數(shù)

  • 執(zhí)行回調(diào)函數(shù)

下面筆者通過一個(gè)簡(jiǎn)單的例子將回調(diào)函數(shù)的實(shí)現(xiàn)與這三部分關(guān)聯(lián)起來。

定義回調(diào)函數(shù)

回調(diào)函數(shù)的定義很簡(jiǎn)單,與普通函數(shù)的定義沒有區(qū)別,比如我們定義一個(gè)看門狗計(jì)時(shí)器的回調(diào)函數(shù)如下:

   
  1. /*高層*/

  2. void Watchdog_ExpiredCallback(void)

  3. {

  4. //do something

  5. }

可以看出這就是一個(gè)普通的函數(shù)。

注冊(cè)回調(diào)函數(shù)并執(zhí)行

注冊(cè)回調(diào)函數(shù)筆者在這里給出兩種實(shí)現(xiàn)思路,先是一種比較直觀的:

   
  1. /*底層*/

  2. void Watchdog_Expired(void (*Callback)(void))

  3. {

  4. Callback();

  5. }

可以看到這個(gè)函數(shù)的形參是一個(gè)函數(shù)指針,因此我們也就可以將我們定義的函數(shù)的指針作為函數(shù)傳到當(dāng)前這個(gè)函數(shù),從而實(shí)現(xiàn)在底層調(diào)用高層的代碼。調(diào)用方法也有兩種形式,分別是以下兩種:

   
  1. Watchdog_Expired(Watchdog_ExpiredCallback);

  2. Watchdog_Expired(&Watchdog_ExpiredCallback);

為什么這兩種調(diào)用方式結(jié)果都一致呢,其實(shí)這也就跟數(shù)組的 a&a[0]的關(guān)系是一個(gè)道理,雖然表征的意義不一致,但是其數(shù)值是相等的。注冊(cè)回調(diào)函數(shù)的第二種方法在形式上看著要比第一種要復(fù)雜一點(diǎn),我們先采用如下方式定義一個(gè)函數(shù)指針:

   
  1. typedef void (*Callback)(void);

  2. static Callback WatchdogExpired = NULL;

然后就可以這樣實(shí)現(xiàn)注冊(cè)函數(shù):

   
  1. void

  2. Watchdog_CallbackRegister(void (*Callback)(void))

  3. {

  4. WatchdogExpired = Callback;

  5. }

然后就可以將我們之前定義的函數(shù)進(jìn)行注冊(cè):

   
  1. Watchdog_CallbackRegister(Watchdog_ExpiredCallback);

這里需要注意的是上述的這個(gè)函數(shù)應(yīng)該在系統(tǒng)初始化的時(shí)候,就完成調(diào)用,然后我們就可以在中斷服務(wù)函數(shù)里完成回調(diào)函數(shù)的執(zhí)行了:

   
  1. void watchdog_ISR(void)

  2. {

  3. if (WatchdogExpired != NULL)

  4. {

  5. WatchdogExpired();

  6. }

  7. }

上述便是回調(diào)函數(shù)的一個(gè)簡(jiǎn)單例子,下面筆者將分析回調(diào)函數(shù)在 rtthread 上的一個(gè)應(yīng)用。

RT-Thread 空閑線程的鉤子函數(shù)

我們首先來看 RT-Thread 對(duì)于空閑線程的介紹:

RT-Thread 空閑線程是系統(tǒng)創(chuàng)建的最低優(yōu)先級(jí)的線程,線程狀態(tài)永遠(yuǎn)為就緒態(tài)。當(dāng)系統(tǒng)中無其他就緒線程存在時(shí),調(diào)度器將調(diào)度到空閑線程,它通常是一個(gè)死循環(huán),且永遠(yuǎn)不能被掛起。在空閑線程中也提供了接口來運(yùn)行用戶設(shè)置的鉤子函數(shù),在空閑線程運(yùn)行時(shí)會(huì)調(diào)用該鉤子函數(shù),適合鉤入功耗管理、看門狗喂狗等工作。

在上述介紹中提到空閑線程提供了接口來運(yùn)行用戶設(shè)置的鉤子函數(shù),那這又是基于什么原理呢?首先我們來看空閑函數(shù)是如何設(shè)置鉤子函數(shù),代碼如下:

   
  1. static void (*idle_hook_list[RT_IDEL_HOOK_LIST_SIZE])();


  2. rt_err_t rt_thread_idle_sethook(void (*hook)(void))

  3. {

  4. rt_size_t i;

  5. rt_base_t level;

  6. rt_err_t ret = -RT_EFULL;


  7. /* disable interrupt */

  8. level = rt_hw_interrupt_disable();


  9. for (i = 0; i < RT_IDEL_HOOK_LIST_SIZE; i++)

  10. {

  11. if (idle_hook_list[i] == RT_NULL)

  12. {

  13. idle_hook_list[i] = hook;

  14. ret = RT_EOK;

  15. break;

  16. }

  17. }

  18. /* enable interrupt */

  19. rt_hw_interrupt_enable(level);


  20. return ret;

  21. }

我們可以看到這個(gè)函數(shù)的形參是一個(gè)函數(shù)指針,自然可以想到是用到了回調(diào)函數(shù)的原理。對(duì)于此函數(shù)的實(shí)現(xiàn),我們可以看到是定義了一個(gè)全局的鉤子函數(shù)數(shù)組,也就是說可以注冊(cè)多個(gè)回調(diào)函數(shù),然后會(huì)根據(jù)注冊(cè)的先后順序進(jìn)行執(zhí)行。既然可以注冊(cè)回調(diào)函數(shù)了,那么我們就可以在應(yīng)用層定義一個(gè)回調(diào)函數(shù),這里以看門狗喂狗為例,實(shí)現(xiàn)代碼如下:

   
  1. static void idle_hook(void)

  2. {

  3. /*喂狗操作*/

  4. rt_device_control(wdg_dev,RT_DEVICE_CTRL_WDT_KEEPALIVE, NULL);

  5. rt_kprintf("feed the dog!\n");

  6. }

定義了回調(diào)函數(shù),我們就可以在主程序里將注冊(cè)該回調(diào)函數(shù)了:

   
  1. int main(void)

  2. {

  3. /*省略看門狗設(shè)備的相關(guān)操作*/

  4. rt_thread_idle_sethook(idle_hook);

  5. }

回調(diào)函數(shù)已經(jīng)注冊(cè),何時(shí)會(huì)執(zhí)行呢?對(duì)于當(dāng)前系統(tǒng)而言,當(dāng)當(dāng)前無其他線程運(yùn)行時(shí),切換到空閑線程時(shí)會(huì)運(yùn)行我們注冊(cè)的回調(diào)函數(shù),空閑線程里面的內(nèi)容是這樣的:

   
  1. static void rt_thread_idle_entry(void *parameter)

  2. {

  3. while (1)

  4. {


  5. #ifdef RT_USING_IDLE_HOOK

  6. rt_size_t i;


  7. for (i = 0; i < RT_IDEL_HOOK_LIST_SIZE; i++)

  8. {

  9. if (idle_hook_list[i] != RT_NULL)

  10. {

  11. idle_hook_list[i]();

  12. }

  13. }

  14. #endif


  15. rt_thread_idle_excute();

  16. #ifdef RT_USING_PM

  17. rt_system_power_manager();

  18. #endif

  19. }

  20. }

上述代碼也印證了剛才所說的,注冊(cè)的多個(gè)回調(diào)函數(shù)會(huì)根據(jù)注冊(cè)的順序依次執(zhí)行。最后,回顧空閑線程鉤子函數(shù)的運(yùn)行過程,也就和文章最開始給出的調(diào)用關(guān)系圖相對(duì)應(yīng)起來了。

總結(jié)

在 RT-Thread 中關(guān)于回調(diào)函數(shù)的例子也不止空閑線程鉤子函數(shù)這一個(gè),還有很多,比如調(diào)度器和串口設(shè)備里也有,不過原理都是一樣的,最終實(shí)現(xiàn)的效果也都是能夠使底層調(diào)用高層定義的代碼。

參考資料:

[1] https://www.embedded.com/increasing-code-flexibility-using-callbacks/

[2]https://www.beningo.com/embedded-basics-callback-functions/

您的閱讀是對(duì)我最大的鼓勵(lì),您的建議是對(duì)我最大地提升,歡迎點(diǎn)擊下方圖片進(jìn)入小程序進(jìn)行評(píng)論或者添加筆者微信相互交流,二維碼在公眾號(hào)底部獲取


免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉