多線程是實時操作系統(tǒng)里面最重要的知識點之一,要學習RTOS,多線程是必須(沒錯,是必須)要熟練掌握的內(nèi)容,只有熟練掌握多線程的使用,才能在平時的項目工作里面用好實時操作系統(tǒng)。
關(guān)于多線程的使用和管理,RT-Thread官方提供了比較豐富的文檔作為參考,具體內(nèi)容可以查看以下鏈接:
https://www.rt-thread.org/document/site/programming-manual/thread/thread/
本文是對RT-Thread多線程學習后的總結(jié),并嘗試從如圖所示的以下幾個方面進行總結(jié)。
什么是多線程?
在單片機上學習RT-Thread的多線程之前,要先把“進程”這個概念先放一邊,因為單片機是沒有多進程概念的。單片機運行操作系統(tǒng),不管多少個任務,他們都是多個(或單個)線程之間進行處理這些任務,單片機一般不涉及多進程。
什么是多線程?在哪些情況下要用到多線程?先來舉一個音樂播放器的例子,這個音樂播放器要做以下這些基本的工作:讀取音樂文件并播放、讀取歌詞并顯示、讀取MV文件并播放。
如果這三個基本的工作不用多線程來完成,單片機使用裸機的方式去做這三個工作的話,必然會造成音樂播放卡頓,歌詞顯示不同步,MV視頻播放與音樂不同步。
因為單片機做這三件事情的時候,是Step by Step的,必須完成一件事情之后,再去做下一件事情,這三件事情是有先后順序的,并且不斷循環(huán)重復,如下圖所示。
而如果采用多線程這種方式來完成這個工作,這個過程就變得相對簡單了,比如針對音樂播放器這個場景,可以設(shè)計這幾個線程來處理:音樂文件讀取線程,歌詞文件讀取線程,MV文件讀取線程,音視頻和歌詞顯示線程。
(此處只為舉例描述多線程的概念,不考慮音視頻編解碼的復雜過程,不考慮線程同步,實際上音樂播放器的實現(xiàn)比此處描述更復雜)
音樂文件讀取線程只負責從磁盤讀取音樂文件,歌詞文件讀取線程和MV文件讀取線程也是同樣的道理,它們只做文件讀取工作,而音視頻和歌詞顯示線程,是負責把讀取到的數(shù)據(jù)進行顯示。這幾個線程的工作過程,如下圖所示。
如上圖所示,這幾個任務看上去是“同時”進行的,每個任務都只完成自己的事情,通過多線程,就可以把原本串行完成的任務改為并行完成,大大提高了工作效率。
所以,通俗地對多線程進行理解,就是把一個比較大型的任務,拆分為多個小型的任務,然后通過合理的調(diào)度方式,讓這幾個小型的任務“同時”運行,當這幾個小型任務完成后,大型的任務也隨之完成,這樣可以大大提高任務的完成效率。
多線程的幾種狀態(tài)
對于運行RT-Thread操作系統(tǒng),線程都處于以下五種狀態(tài)的其中一種(初始狀態(tài)、就緒狀態(tài)、運行狀態(tài)、掛起狀態(tài)、關(guān)閉狀態(tài)),通過調(diào)用操作系統(tǒng)提供的接口函數(shù),可以讓線程在這五種狀態(tài)中進行來回切換。
關(guān)于這五種線程狀態(tài)的描述,如下表所示:
多線程的API函數(shù)
如上圖的狀態(tài)機所示,多線程可以通過調(diào)用系統(tǒng)提供的函數(shù)接口,在多個狀態(tài)之間進行切換。這些API函數(shù)在官方提供的參考文檔里面都有詳細的說明描述,以下列舉一些比較常用的函數(shù)接口。
上下滑動查看 API 函數(shù)
多線程的應用示例
多線程的應用示例,主要是為了驗證以上的多線程API接口函數(shù),并且通過實驗現(xiàn)象觀察多線程的運行情況,主要有以下三個示例:
示例源碼下載鏈接:
https://github.com/embediot/rtthread_study_notes
1、線程動態(tài)創(chuàng)建與靜態(tài)創(chuàng)建、線程退出示例。
這個示例主要是通過動態(tài)方式創(chuàng)建線程1,,通過靜態(tài)方式創(chuàng)建線程2,線程1的優(yōu)先級比線程2的優(yōu)先級低,因此可以被線程2搶占。線程2運行10次后就會主動退出,初始化代碼如下圖所示。
2、相同優(yōu)先級線程的時間片輪轉(zhuǎn)調(diào)度示例。
這個示例主要是通過動態(tài)方式創(chuàng)建線程1和線程2,這兩個線程都是相同的優(yōu)先級,并且共用一個線程入口函數(shù),主要是通過傳入不同的線程參數(shù)以區(qū)分線程1和線程2。線程2運行所占用的時間片比線程1要少,因此線程2運行的時間比較短,初始化代碼如下圖所示。
3、線程調(diào)度器的鉤子函數(shù)使用示例。
這個示例主要測試了線程在進行調(diào)度時,關(guān)于鉤子函數(shù)的調(diào)用情況。通過線程調(diào)度器的鉤子函數(shù),打印出線程間的切換信息,初始化的代碼如下圖所示。
多線程應用的注意事項
在使用RT-Thread實時操作系統(tǒng)進行多線程應用開發(fā)的時候,應該要注意以下事項:
1、RT-Thread的線程調(diào)度器是搶占式的,也就是能夠保證就緒隊列里面,最高優(yōu)先級的任務總能獲得CPU的使用權(quán),在任務設(shè)計的時候,要充分考慮好任務的優(yōu)先級。
2、在硬件中斷服務程序運行期間,如果有高優(yōu)先級的任務就緒,那么被中斷的低優(yōu)先級任務將被掛起,高優(yōu)先級的任務將會獲得CPU的使用權(quán)。
3、每個線程都有獨立的線程棧,用來保存線程調(diào)度時上下文的信息,因此在創(chuàng)建線程分配棧空間的時候,要充分考慮棧的大小。
4、在線程的循環(huán)體里面,應該要設(shè)置某些條件,在必要的時候主動讓出CPU的使用權(quán),特別對于高優(yōu)先級的線程,如果程序里面有死循環(huán)操作而又不主動讓出CPU使用權(quán),那么這個線程將會一直占用CPU,并且低優(yōu)先級的線程永遠不會被調(diào)度執(zhí)行。
5、對于沒有一直循環(huán)執(zhí)行的線程,線程執(zhí)行完畢后,資源的回收情況實際上是在空閑線程里面進行的,線程變?yōu)殛P(guān)閉狀態(tài)后,不代表資源馬上被回收。
6、系統(tǒng)空閑線程是最低優(yōu)先級且永遠為就緒狀態(tài)的,空閑線程是一個死循環(huán),永遠不會被掛起,但可以被其他高優(yōu)先級任務搶占,空閑線程主要執(zhí)行僵尸線程的資源回收工作。
7、空閑線程也可以設(shè)置鉤子函數(shù),用來進行功耗管理,看門狗喂狗等工作。
8、通過動態(tài)方式創(chuàng)建的線程,需要設(shè)置好系統(tǒng)堆內(nèi)存的大小,而通過靜態(tài)方式創(chuàng)建的線程,線程棧和線程句柄在程序編譯的時候就已經(jīng)確定,不能被動態(tài)分配,也不能被釋放。
9、大多數(shù)線程都是在不斷循環(huán)執(zhí)行的,無需進行刪除,一般不推薦主動刪除線程。線程運行完畢后,系統(tǒng)調(diào)度器將會自動把線程加入僵尸隊列,資源回收工作將在空閑線程里面進行。
責任編輯:lq
-
操作系統(tǒng)
+關(guān)注
關(guān)注
37文章
7103瀏覽量
125041 -
多線程
+關(guān)注
關(guān)注
0文章
279瀏覽量
20322 -
RT-Thread
+關(guān)注
關(guān)注
32文章
1372瀏覽量
41557
原文標題:【學習筆記】RT-Thread 多線程學習總結(jié)
文章出處:【微信號:RTThread,微信公眾號:RTThread物聯(lián)網(wǎng)操作系統(tǒng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
RT-Thread Nano移植后動態(tài)創(chuàng)建線程創(chuàng)建不了怎么處理?
RT-Thread審核團招募: 深度參與開源RTOS社區(qū)治理與演進

請問rt-thread studio如何進行多線程編譯?
如何將RT-Thread移植到NXP MCUXPressoIDE上

RT-Thread上CAN實踐

【S32K146 RT-thread】之 SPI驅(qū)動適配

開源共生 商業(yè)共贏 | RT-Thread 2024開發(fā)者大會報名啟動!

【成都】9月21日RT-Thread巡回線下培訓-OpenMV機器視覺

【大連】9月7日RT-Thread巡回線下培訓-OpenMV機器視覺

【武漢】9月7日RT-Thread巡回線下培訓-RTduino-10分鐘上手嵌入式

【QEMU系列】不用開發(fā)板運行RT-Thread指南-ARM架構(gòu)

2024 RT-Thread全球巡回 線下培訓火熱來襲!

【好書推薦】RT-Thread設(shè)備驅(qū)動開發(fā)指南

RT-Thread 新里程碑達成——GitHub Star 破萬!

基于 RT-Thread專業(yè)版的EtherCAT主站方案

評論