0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

線程的基本知識(shí)

黃工的嵌入式技術(shù)圈 ? 來(lái)源:黃工的嵌入式技術(shù)圈 ? 作者:黃工的嵌入式技術(shù) ? 2020-02-04 15:42 ? 次閱讀

今天給大家分享一點(diǎn)RT-Thread的基礎(chǔ)知識(shí)。

什么是線程?

人們?cè)谏钪刑幚韽?fù)雜問(wèn)題時(shí),慣用的方法就是分而治之,即把一個(gè)大問(wèn)題分解成多個(gè)相對(duì)簡(jiǎn)單、比較容易解決的小問(wèn)題,小問(wèn)題逐個(gè)被解決了,大問(wèn)題也就隨之解決了。同樣,在設(shè)計(jì)一個(gè)較為復(fù)雜的應(yīng)用程序時(shí),也通常把一個(gè)大型任務(wù)分解成多個(gè)小任務(wù),然后通過(guò)運(yùn)行這些小任務(wù),最終達(dá)到完成大任務(wù)的目的。

在裸機(jī)系統(tǒng)中, 系統(tǒng)的主體就是 main 函數(shù)里面順序執(zhí)行的無(wú)限循環(huán),這個(gè)無(wú)限循環(huán)里面 CPU 按照順序完成各種事情。在多線程系統(tǒng)中,我們根據(jù)功能的不同,把整個(gè)系統(tǒng)分割成一個(gè)個(gè)獨(dú)立的且無(wú)法返回的函數(shù),這個(gè)函數(shù)我們稱為線程。

線程由哪些部分組成?

RT-Thread 中的線程由三部分組成:線程代碼(函數(shù))、線程控制塊、線程堆棧。

線程棧

在一個(gè)裸機(jī)系統(tǒng)中, 如果有全局變量,有子函數(shù)調(diào)用,有中斷發(fā)生。那么系統(tǒng)在運(yùn)行的時(shí)候,全局變量放在哪里,子函數(shù)調(diào)用時(shí),局部變量放在哪里, 中斷發(fā)生時(shí),函數(shù)返回地址發(fā)哪里。

如果只是單純的裸機(jī)編程,它們放哪里我們不用管,但是如果要寫一個(gè) RTOS,這些種種環(huán)境參數(shù),我們必須弄清楚他們是如何存儲(chǔ)的。

在裸機(jī)系統(tǒng)中,他們統(tǒng)統(tǒng)放在一個(gè)叫棧的地方,棧是單片機(jī) RAM 里面一段連續(xù)的內(nèi)存空間,棧的大小一般在啟動(dòng)文件或者鏈接腳本里面指定, 最后由 C 庫(kù)函數(shù)_main 進(jìn)行初始化。

但是, 在多線程系統(tǒng)中,每個(gè)線程都是獨(dú)立的,互不干擾的,所以要為每個(gè)線程都分配獨(dú)立的??臻g,這個(gè)??臻g通常是一個(gè)預(yù)先定義好的全局?jǐn)?shù)組, 也可以是動(dòng)態(tài)分配的一段內(nèi)存空間,但它們都存在于 RAM 中。如:

staticrt_uint8_tled_stack[512];

線程棧其實(shí)就是一個(gè)預(yù)先定義好的全局?jǐn)?shù)據(jù),數(shù)據(jù)類型為rt_uint8_t,大小我們?cè)O(shè)置為 512。在 RT-Thread 中,凡是涉及到數(shù)據(jù)類型的地方, RTThread 都會(huì)將標(biāo)準(zhǔn)的 C 數(shù)據(jù)類型用 typedef 重新取一個(gè)類型名, 以“rt”前綴開頭。這些經(jīng)過(guò)重定義的數(shù)據(jù)類型放在 rtdef.h ,如:

線程控制塊

在 RT-Thread 中,線程控制塊由結(jié)構(gòu)體 struct rt_thread 表示,線程控制塊是操作系統(tǒng)用于管理線程的一個(gè)數(shù)據(jù)結(jié)構(gòu),它會(huì)存放線程的一些信息,例如優(yōu)先級(jí)、線程名稱、線程狀態(tài)等,也包含線程與線程之間連接用的鏈表結(jié)構(gòu),線程等待事件集合等,詳細(xì)定義如下(在rtdef.h中定義):

為led線程定義一個(gè)線程控制塊:

staticstructrt_threadled_thread;

線程函數(shù)

線程控制塊中的 entry 是線程的入口函數(shù),它是線程實(shí)現(xiàn)預(yù)期功能的函數(shù)。線程的入口函數(shù)由用戶設(shè)
計(jì)實(shí)現(xiàn),一般有以下兩種代碼形式:

無(wú)限循環(huán)模式:

在實(shí)時(shí)系統(tǒng)中,線程通常是被動(dòng)式的:這個(gè)是由實(shí)時(shí)系統(tǒng)的特性所決定的,實(shí)時(shí)系統(tǒng)通??偸堑却饨缡录陌l(fā)生,而后進(jìn)行相應(yīng)的服務(wù):

順序執(zhí)行或有限次循環(huán)模式:

如簡(jiǎn)單的順序語(yǔ)句、 do whlie() 或 for() 循環(huán)等,此類線程不會(huì)循環(huán)或不會(huì)永久循環(huán),可謂是 “一次性”線程,一定會(huì)被執(zhí)行完畢。在執(zhí)行完畢后,線程將被系統(tǒng)自動(dòng)刪除。

動(dòng)態(tài)線程與靜態(tài)線程

我們的用戶線程有兩種創(chuàng)建方式,一種是靜態(tài)線程,另一種是動(dòng)態(tài)線程。

創(chuàng)建靜態(tài)線程的函數(shù):

返回值為錯(cuò)誤代碼。

創(chuàng)建動(dòng)態(tài)線程的函數(shù):

返回值為線程控制塊 。

線程創(chuàng)建實(shí)例

創(chuàng)建一個(gè)靜態(tài)線程

1、確定線程棧

2、定義線程控制塊

3、創(chuàng)建線程函數(shù)。

#include #include #include /*靜態(tài)線程相關(guān)宏定義*/ #defineTHREAD_PRIORITY25/*優(yōu)先級(jí)*/ #defineSTACK_SIZE512/*棧大小*/ #defineTIMESLICE5/*時(shí)間片*/ /*線程三要素*/ staticrt_uint8_tstatic_thread_stack[STACK_SIZE];/*線程棧*/ staticstructrt_threadstatic_thread;/*線程控制塊*/ staticvoidstatic_thread_entry(void*parameter);/*線程入口函數(shù)*/ /*靜態(tài)線程入口函數(shù)*/ staticvoidstatic_thread_entry(void*parameter) { rt_uint32_ti=0; rt_kprintf("Thisisstaticthread!\n"); /*無(wú)限循環(huán)*/ while(1) { rt_kprintf("staticthreadcount:%d\r\n",++i); /*等待0.5s,讓出cpu權(quán)限,切換到其他線程*/ rt_thread_delay(500); } } /*主函數(shù)*/ intmain(void) { rt_err_tresult; /*創(chuàng)建靜態(tài)線程:優(yōu)先級(jí) 25 ,時(shí)間片 5個(gè)系統(tǒng)滴答,線程棧512字節(jié)*/ result=rt_thread_init(&static_thread, "static_thread", static_thread_entry, RT_NULL, (rt_uint8_t*)&static_thread_stack[0], STACK_SIZE, THREAD_PRIORITY, TIMESLICE); /*創(chuàng)建成功則啟動(dòng)靜態(tài)線程*/ if(result==RT_EOK) { rt_thread_startup(&static_thread); } }

運(yùn)行結(jié)果為:

可見,在T-Thread中創(chuàng)建一個(gè)線程需要線程棧、線程控制塊與線程函數(shù)這三要素。除此之外,需要設(shè)置一個(gè)線程優(yōu)先級(jí),因?yàn)镽T-Thread的調(diào)度器是基于優(yōu)先級(jí)的搶占式調(diào)度算法。還需要設(shè)置一個(gè)時(shí)間片參數(shù),這個(gè)用于多個(gè)線程具有同等優(yōu)先級(jí)的情況下,采用時(shí)間片的輪轉(zhuǎn)調(diào)度算法進(jìn)行調(diào)度,這個(gè)值與時(shí)間節(jié)拍有關(guān),每一秒的節(jié)拍數(shù)可在rtconfig.h里進(jìn)行設(shè)置:

在這里我們只創(chuàng)建一個(gè)線程,所以時(shí)間片我們沒有用到,但也需要傳遞一個(gè)時(shí)間片的值給rt_thread_init函數(shù)。最后,在主函數(shù)里調(diào)用相關(guān)接口創(chuàng)建一個(gè)靜態(tài)線程,創(chuàng)建成功則啟動(dòng)該線程。

創(chuàng)建一個(gè)動(dòng)態(tài)線程

創(chuàng)建動(dòng)態(tài)線程與創(chuàng)建靜態(tài)線程類似:

#include #include #include /*動(dòng)態(tài)線程相關(guān)宏定義*/ #defineTHREAD_PRIORITY25/*優(yōu)先級(jí)*/ #defineSTACK_SIZE512/*棧大小*/ #defineTIMESLICE5/*時(shí)間片*/ /*線程三要素*/ staticrt_uint8_tdynamic_thread_stack[STACK_SIZE];/*線程棧*/ staticstructrt_threaddynamic_thread;/*線程控制塊*/ staticvoiddynamic_thread_entry(void*parameter);/*線程入口函數(shù)*/ /*動(dòng)態(tài)線程入口函數(shù)*/ staticvoiddynamic_thread_entry(void*parameter) { rt_uint32_ti; /*無(wú)限循環(huán)*/ while(1) { for(i=0;i

運(yùn)行結(jié)果:

靜態(tài)線程VS動(dòng)態(tài)線程

上例中,從運(yùn)行結(jié)果上看,是沒有任何差別的!那么,我們?cè)趯?shí)際中如何抉擇?

使用靜態(tài)線程時(shí),必須先定義靜態(tài)的線程控制塊,并且定義好棧空間,然后調(diào)用rt_thread_init()函數(shù)來(lái)完成線程的初始化工作。采用這種方式,線程控制塊和堆棧占用的內(nèi)存會(huì)放在 RW/ZI 段,這段空間在編譯時(shí)就已經(jīng)確定,它不是可以動(dòng)態(tài)分配的,所以不能被釋放,而只能使用 rt_thread_detach()函數(shù)將該線程控制塊從對(duì)象管理器中脫離。
使用動(dòng)態(tài)定義方式 rt_thread_create()時(shí), RT-Thread 會(huì)動(dòng)態(tài)申請(qǐng)線程控制塊和堆棧空間。在編譯時(shí),編譯器是不會(huì)感知到這段空間的,只有在程序運(yùn)行時(shí), RT-Thread 才會(huì)從系統(tǒng)堆中申請(qǐng)分配這段內(nèi)存空間,當(dāng)不需要使用該線程時(shí),調(diào)用 rt_thread_delete()函數(shù)就會(huì)將這段申請(qǐng)的內(nèi)存空間重新釋放到內(nèi)存堆中。

這兩種方式各有利弊,靜態(tài)定義方式會(huì)占用 RW/ZI 空間,但是不需要?jiǎng)討B(tài)分配內(nèi)存,運(yùn)行時(shí)效率較高,實(shí)時(shí)性較好。動(dòng)態(tài)方式不會(huì)占用額外的 RW/ZI 空間,占用空間小,但是運(yùn)行時(shí)需要?jiǎng)討B(tài)分配內(nèi)存,效率沒有靜態(tài)方式高。

總的來(lái)說(shuō),這兩種方式就是空間和時(shí)間效率的平衡,可以根據(jù)實(shí)際環(huán)境需求選擇采用具體的分配方式。就像C編程中,何時(shí)使用動(dòng)態(tài)空間,何時(shí)使用靜態(tài)空間,也需要根據(jù)實(shí)際情況平衡選擇。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 多線程
    +關(guān)注

    關(guān)注

    0

    文章

    276

    瀏覽量

    19878
  • 線程
    +關(guān)注

    關(guān)注

    0

    文章

    502

    瀏覽量

    19613
  • RT-Thread
    +關(guān)注

    關(guān)注

    31

    文章

    1249

    瀏覽量

    39719
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    CPU線程和程序線程的區(qū)別

    CPU的線程與程序的線程在概念、作用、實(shí)現(xiàn)方式以及性能影響等方面存在顯著差異。以下是對(duì)兩者區(qū)別的詳細(xì)闡述,旨在深入探討這一技術(shù)話題。
    的頭像 發(fā)表于 09-02 11:18 ?543次閱讀

    堆棧和內(nèi)存的基本知識(shí)

    本文主要聊聊關(guān)于堆棧的內(nèi)容。包括堆棧和內(nèi)存的基本知識(shí)。常見和堆棧相關(guān)的 bug,如棧溢出,內(nèi)存泄漏,堆內(nèi)存分配失敗等。后面介紹軟件中堆棧統(tǒng)計(jì)的重要性,以及如何使用工具工具軟件中堆棧使用的范圍,并給出在軟件開發(fā)中,如何降低堆棧問(wèn)題,優(yōu)化堆棧的一些實(shí)踐。
    的頭像 發(fā)表于 08-29 14:10 ?320次閱讀
    堆棧和內(nèi)存的<b class='flag-5'>基本知識(shí)</b>

    光刻工藝的基本知識(shí)

    在萬(wàn)物互聯(lián),AI革命興起的今天,半導(dǎo)體芯片已成為推動(dòng)現(xiàn)代社會(huì)進(jìn)步的心臟。而光刻(Lithography)技術(shù),作為先進(jìn)制造中最為精細(xì)和關(guān)鍵的工藝,不管是半導(dǎo)體芯片、MEMS器件,還是微納光學(xué)元件都離不開光刻工藝的參與,其重要性不言而喻。本文將帶您一起認(rèn)識(shí)光刻工藝的基本知識(shí)。
    的頭像 發(fā)表于 08-26 10:10 ?431次閱讀
    光刻工藝的<b class='flag-5'>基本知識(shí)</b>

    熱釋電傳感器基本知識(shí)和使用中的常見問(wèn)題

    電子發(fā)燒友網(wǎng)站提供《熱釋電傳感器基本知識(shí)和使用中的常見問(wèn)題》資料免費(fèi)下載
    發(fā)表于 06-11 09:06 ?1次下載

    什么是動(dòng)態(tài)線程池?動(dòng)態(tài)線程池的簡(jiǎn)單實(shí)現(xiàn)思路

    因此,動(dòng)態(tài)可監(jiān)控線程池一種針對(duì)以上痛點(diǎn)開發(fā)的線程池管理工具。主要可實(shí)現(xiàn)功能有:提供對(duì) Spring 應(yīng)用內(nèi)線程池實(shí)例的全局管控、應(yīng)用運(yùn)行時(shí)動(dòng)態(tài)變更線程池參數(shù)以及
    的頭像 發(fā)表于 02-28 10:42 ?529次閱讀

    低壓配電基本知識(shí)

    低壓配電基本知識(shí)
    的頭像 發(fā)表于 12-21 09:48 ?711次閱讀
    低壓配電<b class='flag-5'>基本知識(shí)</b>

    redis多線程還能保證線程安全嗎

    Redis是一種使用C語(yǔ)言編寫的高性能鍵值存儲(chǔ)系統(tǒng),它是單線程的,因?yàn)槭褂昧硕嗦窂?fù)用的方式來(lái)處理并發(fā)請(qǐng)求。這樣的實(shí)現(xiàn)方式帶來(lái)了很好的性能,但同時(shí)也引發(fā)了一些線程安全方面的問(wèn)題。 在Redis中,由于
    的頭像 發(fā)表于 12-05 10:28 ?1513次閱讀

    電子元器件采購(gòu)基本知識(shí)

    電子元器件是指電子設(shè)備中的各種電子部件,包括電阻、電容、電感、二極管、三極管、集成電路等。在電子設(shè)備的設(shè)計(jì)和制造過(guò)程中,電子元器件的采購(gòu)是非常重要的環(huán)節(jié)。以下是一些電子元器件采購(gòu)的基本知識(shí)。
    的頭像 發(fā)表于 12-05 09:19 ?1517次閱讀

    核心線程數(shù)和最大線程數(shù)怎么設(shè)置

    核心線程數(shù)和最大線程數(shù)是Java線程池中重要的參數(shù),用來(lái)控制線程池中線程的數(shù)量和行為。正確地設(shè)置這兩個(gè)參數(shù)可以優(yōu)化系統(tǒng)的性能和資源利用率。本
    的頭像 發(fā)表于 12-01 13:50 ?8097次閱讀

    西門子伺服驅(qū)動(dòng)器維修的一些基本知識(shí)

    西門子伺服驅(qū)動(dòng)器維修的一些基本知識(shí)
    的頭像 發(fā)表于 11-23 10:55 ?1576次閱讀

    示波器應(yīng)用的基本知識(shí)

    電子發(fā)燒友網(wǎng)站提供《示波器應(yīng)用的基本知識(shí).pdf》資料免費(fèi)下載
    發(fā)表于 11-18 14:50 ?1次下載
    示波器應(yīng)用的<b class='flag-5'>基本知識(shí)</b>

    如何查看一個(gè)線程的ID

    1.什么是線程? linux內(nèi)核中是沒有線程這個(gè)概念的,而是輕量級(jí)進(jìn)程的概念:LWP。一般我們所說(shuō)的線程概念是C庫(kù)當(dāng)中的概念。 1.1線程是怎樣描述的?
    的頭像 發(fā)表于 11-13 14:38 ?1142次閱讀
    如何查看一個(gè)<b class='flag-5'>線程</b>的ID

    Linux線程線程與異步編程、協(xié)程與異步介紹

    協(xié)程不是系統(tǒng)級(jí)線程,很多時(shí)候協(xié)程被稱為“輕量級(jí)線程”、“微線程”、“纖程(fiber)”等。簡(jiǎn)單來(lái)說(shuō)可以認(rèn)為協(xié)程是線程里不同的函數(shù),這些函數(shù)之間可以相互快速切換。 協(xié)程和用戶態(tài)
    的頭像 發(fā)表于 11-11 11:35 ?940次閱讀
    Linux<b class='flag-5'>線程</b>、<b class='flag-5'>線程</b>與異步編程、協(xié)程與異步介紹

    線程池基本概念與原理

    一、線程池基本概念與原理 1.1 線程池概念及優(yōu)勢(shì) C++線程池簡(jiǎn)介 線程池是一種并發(fā)編程技術(shù),它能有效地管理并發(fā)的線程、減少資源占用和提高
    的頭像 發(fā)表于 11-10 10:24 ?444次閱讀

    汽車電子中幾個(gè)重要器件的基本知識(shí)

    電子發(fā)燒友網(wǎng)站提供《汽車電子中幾個(gè)重要器件的基本知識(shí).doc》資料免費(fèi)下載
    發(fā)表于 10-27 14:25 ?2次下載
    汽車電子中幾個(gè)重要器件的<b class='flag-5'>基本知識(shí)</b>