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

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

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

stm32多線程 單片機如何實現(xiàn)多線程

如意 ? 來源:百度百科、與非網(wǎng)、CSDN ? 作者:百度百科、與非網(wǎng) ? 2021-07-22 14:51 ? 次閱讀

stm32多線程單片機如何實現(xiàn)多線程

STM 32系列是專門應(yīng)用在高性能、低成本、低功耗的嵌入式應(yīng)用設(shè)計的ARM Corte-M0,M0+,M3,M4和M7內(nèi)核,是主流的嵌入式單片機之一。

多線程通常是指從計算機軟件或硬件上實現(xiàn)多個線程并發(fā)執(zhí)行的技術(shù),多線程技術(shù)有利于提升計算機整體處理性能。

基于STM32單片機的多線程源代碼實例1:

#include “Hal_Led/Hal_Led.h”

#include “Hal_delay/delay.h”

#include “Hal_Key/Hal_Key.h”

#include “ringbuffer.h”

#define APP_LED2_BLINK_EVENT 0x0001

#define HAL_LED1_BLINK_EVENT 0x0001

#define TASK_NO_TASK_RUNNING 0xFF

unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events );

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events );

typedef unsigned short uint16;

typedef unsigned char uint8;

#define TASK_CNT 2 //定義線程的個數(shù)

//定義函數(shù)指針

typedef unsigned short (*pTaskEventHandlerFn)( unsigned char task_id, unsigned short events );

//線程函數(shù)表

const pTaskEventHandlerFn tasksArr[] =

{

Hal_ProcessEvent,

App_ProcessEvent

};

const unsigned char tasksCnt = sizeof( tasksArr ) / sizeof( tasksArr[ 0] );

//uint16 *tasksEvents;

uint16 tasksEvents[TASK_CNT] = { 0}; //每個線程有16位位域空間用于設(shè)置事件

static uint8 activeTaskID = 0xFF; //當(dāng)前任務(wù)ID,指示作用

#define SUCCESS 0x00

#define FAILURE 0x01

#define INVALID_TASK 0x02

uint8 osal_set_event( uint8 task_id, uint16 event_flag )

{

if ( task_id 《 tasksCnt )

{

tasksEvents[task_id] |= event_flag; // Stuff the event bit(s)

return ( SUCCESS );

}

else

{

return ( INVALID_TASK );

}

}

/**

* @brief 程序入口

* @param none

* @return none

*/

int main( void)

{

unsigned short taskID = 0;

uint8 idx = 0;

SystemInit(); //系統(tǒng)時鐘初始化

delayInit( 72); //滴答定時器初始化

Led_Init(); //LED初始化

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

osal_set_event( 0, HAL_LED1_BLINK_EVENT);

osal_set_event( 1, APP_LED2_BLINK_EVENT);

while( 1)

{

do

{

if(tasksEvents[idx]) //輪訓(xùn)獲知哪個線程有事件需要進行處理

{

break;

}

}

while (++idx 《 tasksCnt);

if (idx 《 tasksCnt)

{

uint16 events;

events = tasksEvents[idx];

tasksEvents[idx] = 0; // 清除事件數(shù)組中的事件

activeTaskID = idx;

events = (tasksArr[idx])( idx, events ); //調(diào)用線程函數(shù)

activeTaskID = TASK_NO_TASK_RUNNING;

tasksEvents[idx] |= events; // 添加未處理的事件到本線程的事件組中

}

delayMs( 1000);

}

}

/**

* @brief 應(yīng)用層處理

* @param none

* @r

*/

unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events )

{

if ( events & HAL_LED1_BLINK_EVENT )

{

Led_Reverse( 1);

return events ^ HAL_LED1_BLINK_EVENT; //清除事件

}

}

/**

* @brief 硬件控制線程

* @param none

* @r

*/

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events )

{

if ( events & APP_LED2_BLINK_EVENT )

{

Led_Reverse( 2);

return events ^ APP_LED2_BLINK_EVENT; //清除事件

}

基于STM32單片機的多線程源代碼實例2:

public class SimpleThread {

//私有的靜態(tài)成員內(nèi)部類,實現(xiàn)了Runnable接口

private static class ThreadMessage implements Runnable{

public void run(){

String[] info = {“消息1”,“消息2”, “消息3”, “消息4”};

try {

for(int i=0;i《info.length;i++){

Thread.sleep(4000);

displayThreadMessage(info[i]);

}

} catch (InterruptedException e) {

displayThreadMessage(“不能正常工作”);

}

}

}

//顯示消息,消息是當(dāng)前線程的名字

static void displayThreadMessage(String message){

String threadName = Thread.currentThread().getName();

//格式化輸出線程消息

System.out.format(“%s: %s%n”, threadName, message);

}

public static void main(String[] args) throws InterruptedException {

// 中斷ThreadMessage線程之前延遲的毫秒數(shù)(默認是一分鐘)

long delay =1000 * 60;

//如果有命令行參數(shù),那么在命令行參數(shù)中給出推遲的時間

if(args.length》0){

try {

delay =Long.parseLong(args[0])*1000;

} catch (NumberFormatException e) {

System.err.println(“參數(shù)必須是整數(shù)”);

System.exit(1);

}

}

displayThreadMessage(“啟動線程ThreadMessage.。。”);

long startTime = System.currentTimeMillis();

Thread t = new Thread(new ThreadMessage());

t.start();

displayThreadMessage(“等待線程ThreadMessage結(jié)速。。?!保?

//循環(huán)直到ThreadMessage線程退出

while(t.isAlive()){

displayThreadMessage(“繼續(xù)等待線程ThreadMessage.。?!保?

//最多等待3秒鐘ThreadMessage線程結(jié)速

t.join(3000);

//如果線程t運行的時間超過delay指定時間

if(((System.currentTimeMillis() - startTime) 》 delay) && t.isAlive()){

displayThreadMessage(“線程ThreadMessage運行時間太久了,不想等待!”);

t.interrupt();

t.join();

}

}

displayThreadMessage(“結(jié)束線程ThreadMessage!??!”);

}

}

基于STM32單片機的多線程源代碼實例3:

#include “Hal_Led/Hal_Led.h”

#include “Hal_delay/delay.h”

#include “Hal_Key/Hal_Key.h”

#include “ringbuffer.h”

#define APP_LED2_BLINK_EVENT 0x0001

#define HAL_LED1_BLINK_EVENT 0x0001

#define TASK_NO_TASK_RUNNING 0xFF

unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events );

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events );

typedef unsigned short uint16;

typedef unsigned char uint8;

#define TASK_CNT 2 // 定義線程的個數(shù)

// 定義函數(shù)指針

typedef unsigned short (*pTaskEventHandlerFn)( unsigned char task_id, unsigned short events );

// 線程函數(shù)表

const pTaskEventHandlerFn tasksArr[] =

{

Hal_ProcessEvent,

App_ProcessEvent

};

const unsigned char tasksCnt = sizeof( tasksArr ) / sizeof( tasksArr[0] );

//uint16 *tasksEvents;

uint16 tasksEvents[TASK_CNT] = {0}; // 每個線程有 16 位位域空間用于設(shè)置事件

staTIc uint8 acTIveTaskID = 0xFF; // 當(dāng)前任務(wù) ID,指示作用

#define SUCCESS 0x00

#define FAILURE 0x01

#define INVALID_TASK 0x02

uint8 osal_set_event( uint8 task_id, uint16 event_flag )

{

if ( task_id 《 tasksCnt )

{

tasksEvents[task_id] |= event_flag; // Stuff the event bit(s)

return ( SUCCESS );

}

else

{

return ( INVALID_TASK );

}

}

/**

* @brief 程序入口

* @param none

* @return none

*/

int main(void)

{

unsigned short taskID = 0;

uint8 idx = 0;

SystemInit(); // 系統(tǒng)時鐘初始化

delayInit(72); // 滴答定時器初始化

Led_Init(); //LED 初始化

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

osal_set_event(0, HAL_LED1_BLINK_EVENT);

osal_set_event(1, APP_LED2_BLINK_EVENT);

while(1)

{

do

{

if(tasksEvents[idx]) // 輪訓(xùn)獲知哪個線程有事件需要進行處理

{

break;

}

}

while (++idx 《 tasksCnt);

if (idx 《 tasksCnt)

{

uint16 events;

events = tasksEvents[idx];

tasksEvents[idx] = 0; // 清除事件數(shù)組中的事件

acTIveTaskID = idx;

events = (tasksArr[idx])( idx, events ); // 調(diào)用線程函數(shù)

activeTaskID = TASK_NO_TASK_RUNNING;

tasksEvents[idx] |= events; // 添加未處理的事件到本線程的事件組中

}

delayMs(1000);

}

}

/**

* @brief 應(yīng)用層處理

* @param none

* @r

*/

unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events )

{

if ( events & HAL_LED1_BLINK_EVENT )

{

Led_Reverse(1);

return events ^ HAL_LED1_BLINK_EVENT; // 清除事件

}

}

/**

* @brief 硬件控制線程

* @param none

* @r

*/

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events )

{

if ( events & APP_LED2_BLINK_EVENT )

{

Led_Reverse(2);

return events ^ APP_LED2_BLINK_EVENT; // 清除事件

以上是關(guān)于STM32單片機的源代碼,希望對用戶有所幫助。

本文整合自百度百科、與非網(wǎng)、CSDN-辣條boy

責(zé)編AJX

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

    關(guān)注

    6023

    文章

    44376

    瀏覽量

    628347
  • STM32
    +關(guān)注

    關(guān)注

    2257

    文章

    10828

    瀏覽量

    352468
  • 多線程
    +關(guān)注

    關(guān)注

    0

    文章

    275

    瀏覽量

    19850
收藏 人收藏

    評論

    相關(guān)推薦

    多線程設(shè)計模式到對 CompletableFuture 的應(yīng)用

    最近在開發(fā) 延保服務(wù) 頻道頁時,為了提高查詢效率,使用到了多線程技術(shù)。為了對多線程方案設(shè)計有更加充分的了解,在業(yè)余時間讀完了《圖解 Java 多線程設(shè)計模式》這本書,覺得收獲良多。本篇文章將介紹其中
    的頭像 發(fā)表于 06-26 14:18 ?180次閱讀
    從<b class='flag-5'>多線程</b>設(shè)計模式到對 CompletableFuture 的應(yīng)用

    java實現(xiàn)多線程的幾種方式

    Java實現(xiàn)多線程的幾種方式 多線程是指程序中包含了兩個或以上的線程,每個線程都可以并行執(zhí)行不同的任務(wù)或操作。Java中的
    的頭像 發(fā)表于 03-14 16:55 ?434次閱讀

    AT socket可以多線程調(diào)用嗎?

    請問AT socket 可以多線程調(diào)用嗎? 有互鎖機制嗎,還是要自己做互鎖。
    發(fā)表于 03-01 08:22

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

    Redis是一種使用C語言編寫的高性能鍵值存儲系統(tǒng),它是單線程的,因為使用了多路復(fù)用的方式來處理并發(fā)請求。這樣的實現(xiàn)方式帶來了很好的性能,但同時也引發(fā)了一些線程安全方面的問題。 在Redis中,由于
    的頭像 發(fā)表于 12-05 10:28 ?1346次閱讀

    mfc多線程編程實例

    (圖形用戶界面)應(yīng)用程序的開發(fā)。在這篇文章中,我們將重點介紹MFC中的多線程編程。 多線程編程在軟件開發(fā)中非常重要,它可以實現(xiàn)程序的并發(fā)執(zhí)行,提高程序的效率和響應(yīng)速度。MFC提供了豐富的多線程
    的頭像 發(fā)表于 12-01 14:29 ?1143次閱讀

    多線程如何保證數(shù)據(jù)的同步

    多線程編程是一種并發(fā)編程的方法,意味著程序中同時運行多個線程,每個線程可獨立執(zhí)行不同的任務(wù),共享同一份數(shù)據(jù)。由于多線程并發(fā)執(zhí)行的特點,會引發(fā)數(shù)據(jù)同步的問題,即保證多個
    的頭像 發(fā)表于 11-17 14:22 ?891次閱讀

    多線程并發(fā)查詢oracle數(shù)據(jù)庫

    數(shù)據(jù)庫的原理、使用場景、實現(xiàn)方法以及可能遇到的問題和解決方案。 一、多線程并發(fā)查詢的原理 在傳統(tǒng)的單線程查詢方式中,當(dāng)一個查詢請求發(fā)起時,數(shù)據(jù)庫會按照順序執(zhí)行查詢語句并返回結(jié)果。如果查詢語句比較復(fù)雜或者數(shù)據(jù)量比較大,查詢的時
    的頭像 發(fā)表于 11-17 14:22 ?3097次閱讀

    多線程同步的幾種方法

    多線程同步是指在多個線程并發(fā)執(zhí)行的情況下,為了保證線程執(zhí)行的正確性和一致性,需要采用特定的方法來協(xié)調(diào)線程之間的執(zhí)行順序和共享資源的訪問。下面將介紹幾種常見的
    的頭像 發(fā)表于 11-17 14:16 ?960次閱讀

    Linux系統(tǒng)上多線程和多進程的運行效率

    關(guān)于多進程和多線程,教科書上最經(jīng)典的一句話是“進程是資源分配的最小單位,線程是CPU調(diào)度的最小單位”,這句話應(yīng)付考試基本上夠了,但如果在工作中遇到類似的選擇問題,那就沒有這么簡單了,選的不好,會讓你
    的頭像 發(fā)表于 11-10 10:54 ?1059次閱讀
    Linux系統(tǒng)上<b class='flag-5'>多線程</b>和多進程的運行效率

    基于DSP/BIOS多線程的小型組合導(dǎo)航系統(tǒng)設(shè)計

    電子發(fā)燒友網(wǎng)站提供《基于DSP/BIOS多線程的小型組合導(dǎo)航系統(tǒng)設(shè)計.pdf》資料免費下載
    發(fā)表于 11-08 10:07 ?0次下載
    基于DSP/BIOS<b class='flag-5'>多線程</b>的小型組合導(dǎo)航系統(tǒng)設(shè)計

    關(guān)于Python多進程和多線程詳解

    進程(process)和線程(thread)是操作系統(tǒng)的基本概念,但是它們比較抽象,不容易掌握。關(guān)于多進程和多線程,教科書上最經(jīng)典的一句話是“進程是資源分配的最小單位,線程是CPU調(diào)度的最小單位”。
    的頭像 發(fā)表于 11-06 14:46 ?687次閱讀
    關(guān)于Python多進程和<b class='flag-5'>多線程</b>詳解

    多線程idm下載軟件

    多線程idm下載軟件
    發(fā)表于 10-23 09:23 ?0次下載

    請問單片機怎么實現(xiàn)真正的多線程?

    單片機怎么實現(xiàn)真正的多線程??
    發(fā)表于 10-18 06:45

    MDK可以支持多線程編譯嗎?

    怎么才能打開多線程編譯
    發(fā)表于 10-11 07:23

    Java多線程的用法

    能力。 什么是進程 是指正在運行的程序的實例。 每個進程都擁有自己的內(nèi)存空間、代碼、數(shù)據(jù)和文件等資源,可以獨立運行、調(diào)度和管理。在操作系統(tǒng)中,進程是系統(tǒng)資源分配的最小單位,是實現(xiàn)多任務(wù)的基礎(chǔ)。 Java多線程 Java多線程是指
    的頭像 發(fā)表于 09-30 17:07 ?843次閱讀