加入星計(jì)劃,您可以享受以下權(quán)益:

  • 創(chuàng)作內(nèi)容快速變現(xiàn)
  • 行業(yè)影響力擴(kuò)散
  • 作品版權(quán)保護(hù)
  • 300W+ 專業(yè)用戶
  • 1.5W+ 優(yōu)質(zhì)創(chuàng)作者
  • 5000+ 長(zhǎng)期合作伙伴
立即加入
  • 正文
    • 一個(gè)延時(shí)的問(wèn)題
    • 相對(duì)延時(shí)和絕對(duì)延時(shí)的含義
    • 相對(duì)延時(shí)和絕對(duì)延時(shí)區(qū)別
    • 換一種方式看區(qū)別
  • 相關(guān)推薦
  • 電子產(chǎn)業(yè)圖譜
申請(qǐng)入駐 產(chǎn)業(yè)圖譜

嵌入式 | RTOS相對(duì)延時(shí)和絕對(duì)延時(shí)的區(qū)別

2022/08/12
1632
閱讀需 6 分鐘
加入交流群
掃碼加入
獲取工程師必備禮包
參與熱點(diǎn)資訊討論

作者 | strongerHuang

微信公眾號(hào) | strongerHuang

嵌入式軟件代碼中延時(shí)是很常見(jiàn)的,只是延時(shí)種類有很多,看你用什么延時(shí)。

一個(gè)延時(shí)的問(wèn)題

問(wèn)題:周期性(固定一個(gè)時(shí)間)去處理某一件事情。你會(huì)通過(guò)什么方式去實(shí)現(xiàn)?

比如:間隔10ms去采集傳感器的數(shù)據(jù),然后通過(guò)一種算法計(jì)算出一個(gè)結(jié)果,最后通過(guò)串口發(fā)送出去。

可能對(duì)于很多習(xí)慣裸機(jī)編程的讀者,首先想到的是:利用定時(shí)器,定時(shí)10ms中斷,在中斷里面處理。

中斷函數(shù)適合處理簡(jiǎn)單數(shù)據(jù),不適合算法、通信等需要長(zhǎng)時(shí)間占用CPU的處理。

對(duì)計(jì)時(shí)精度要求比較高的地方適合定時(shí)器,像本章節(jié)說(shuō)的周期性采集傳感器數(shù)據(jù),要求不適合很高,那么就引入本文說(shuō)的絕對(duì)延時(shí)。

實(shí)時(shí)操作系統(tǒng)FreeRTOS任務(wù)中,利用vTaskDelayUntil絕對(duì)延時(shí)即可完美解決這個(gè)問(wèn)題。

相對(duì)延時(shí)和絕對(duì)延時(shí)的含義

本文拿FreeRTOS中相對(duì)延時(shí)函數(shù)vTaskDelay,絕對(duì)延時(shí)函數(shù)vTaskDelayUntil來(lái)說(shuō)明。

相對(duì)延時(shí):指每次延時(shí)都是從執(zhí)行函數(shù)vTaskDelay()開(kāi)始,直到延時(shí)指定的時(shí)間(參數(shù):滴答值)結(jié)束。

絕對(duì)延時(shí):指每隔指定的時(shí)間(參數(shù):滴答值),執(zhí)行一次調(diào)用vTaskDelayUntil()函數(shù)的任務(wù)。

文字描述可能不夠直觀理解,下面章節(jié)結(jié)合代碼例子、延時(shí)值(IO高低變化波形)、任務(wù)執(zhí)行圖來(lái)詳細(xì)講述一下他們的區(qū)別。

相對(duì)延時(shí)和絕對(duì)延時(shí)區(qū)別

以實(shí)際代碼為例說(shuō)明:一個(gè)任務(wù)中,添加一個(gè)10ms系統(tǒng)延時(shí),然后,在執(zhí)行任務(wù)(耗時(shí)1ms左右,例子以延時(shí)代替)。
相對(duì)延時(shí)代碼:

絕對(duì)延時(shí)代碼:

說(shuō)明:

1.TestDelay這個(gè)延時(shí)函數(shù)僅僅用于測(cè)試(延時(shí)1ms),用于代替采集、算法、發(fā)送等耗時(shí)時(shí)間。

2.兩個(gè)代碼唯一區(qū)別在于系統(tǒng)延時(shí)不同,一個(gè)vTaskDelay(10);,一個(gè)vTaskDelayUntil(&xLastWakeTime, 10);

3.系統(tǒng)時(shí)鐘頻率為1000,也就是上面系統(tǒng)延時(shí)10個(gè)滴答,即10ms。

看到代碼,你想到了他們輸出結(jié)果的差異嗎?來(lái)看下結(jié)果的差異:用PA0這個(gè)引腳輸出的高低電平,得出延時(shí)時(shí)間。
相對(duì)延時(shí)結(jié)果:

絕對(duì)延時(shí)結(jié)果:

結(jié)果為:相對(duì)延時(shí)的周期為系統(tǒng)延時(shí)10ms + 執(zhí)行任務(wù)1ms的時(shí)間,總共11ms時(shí)間。絕對(duì)延時(shí)的周期即為10ms時(shí)間.

換一種方式看區(qū)別

如果上面的區(qū)別還沒(méi)明白,再來(lái)講一個(gè)更容易理解的區(qū)別,通過(guò)文字 + 任務(wù)執(zhí)行圖來(lái)說(shuō)明。

1.相對(duì)延時(shí)先看任務(wù)執(zhí)行圖,按照上面代碼的方式呈現(xiàn):

這里會(huì)牽涉到操作系統(tǒng)任務(wù)切換、高優(yōu)先級(jí)任務(wù)搶占等一些原理,若不了解,請(qǐng)轉(zhuǎn)移直到了解再回來(lái)。

上電,TEST任務(wù)進(jìn)入延時(shí)(阻塞)狀態(tài),此時(shí)系統(tǒng)執(zhí)行其他就緒任務(wù)。FreeRTOS內(nèi)核會(huì)周期性的檢查TEST任務(wù)的阻塞是否達(dá)到,如果阻塞時(shí)間達(dá)到,則將TEST任務(wù)設(shè)置為就緒狀態(tài),如果就緒任務(wù)中TEST任務(wù)的優(yōu)先級(jí)最高,則會(huì)搶占CPU,再次執(zhí)行任務(wù)主體代碼,不斷循環(huán)。

TEST任務(wù)每次系統(tǒng)延時(shí)都是從調(diào)用延時(shí)函數(shù)vTaskDelay()開(kāi)始算起的,所以叫相對(duì)延時(shí)。

從上圖可以看出:如果執(zhí)行TEST任務(wù)的過(guò)程中發(fā)生中斷,或者具有更高優(yōu)先級(jí)的任務(wù)搶占了,那么TEST任務(wù)執(zhí)行的周期就會(huì)變長(zhǎng),所以使用相對(duì)延時(shí)函數(shù)vTaskDelay(),不能周期性的執(zhí)行TEST任務(wù)。

2.絕對(duì)延時(shí)

代碼中定義的變量xLastWakeTime,其實(shí)是用來(lái)保存上一次的系統(tǒng)計(jì)數(shù)器值(方便檢測(cè)下一個(gè)延時(shí)時(shí)間是否到來(lái))。

和上面相對(duì)延時(shí)程序執(zhí)行圖比較,可以看出,系統(tǒng)延時(shí)的時(shí)間包含了程序執(zhí)行的時(shí)間。即時(shí)中途有中斷,或更高優(yōu)先級(jí)任務(wù)打斷,不會(huì)影響下一次執(zhí)行的時(shí)間(也就是這個(gè)周期不會(huì)變,當(dāng)然,打斷時(shí)間不能超過(guò)系統(tǒng)延時(shí)值)。

提示:圖片中添加了一段話:一般來(lái)說(shuō),程序執(zhí)行時(shí)間要小于總間隔時(shí)間(10ms)。

如果打斷時(shí)間太長(zhǎng),回來(lái)之后延時(shí)都超過(guò)了,則會(huì)立馬執(zhí)行程序,不會(huì)再延時(shí)(任務(wù)不會(huì)再阻塞延時(shí))。

相關(guān)推薦

電子產(chǎn)業(yè)圖譜

作者黃工,從事嵌入式軟件開(kāi)發(fā)工作8年有余,高級(jí)嵌入式軟件工程師,業(yè)余維護(hù)公眾號(hào)『strongerHuang』,分享嵌入式軟硬件、單片機(jī)、物聯(lián)網(wǎng)等內(nèi)容。