找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 1225|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

RT_thread系統(tǒng)下通過消息隊(duì)列傳輸數(shù)組實(shí)現(xiàn)線程間通信

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:313887 發(fā)表于 2023-12-26 23:42 | 只看該作者 回帖獎勵 |倒序?yàn)g覽 |閱讀模式
對于一個數(shù)據(jù)采集線程,和一個數(shù)據(jù)接收線程而言。消息隊(duì)列相較于郵箱能夠傳輸不定長字節(jié)的數(shù)據(jù)。但是,對于數(shù)據(jù)采集線程,有時(shí)需要持續(xù)性采集多次的數(shù)據(jù),每次都放進(jìn)數(shù)組中,然后將他們一次性全部發(fā)送出去,這個時(shí)候就是需要注意
區(qū)分,我們到底是每次采完直接發(fā)送,,達(dá)到指定數(shù)據(jù)后才提醒接收線程接收。
還是將數(shù)據(jù)放在數(shù)組中,將數(shù)組一次性發(fā)出。
舉個簡單的例子,比如,線程1需要采集50個數(shù)據(jù)為1組發(fā)送給線程2
此時(shí),線程1每接收一個發(fā)送一個,然后線程2統(tǒng)一接收。
#include <rtthread.h>
#define ARRAY_SIZE 50
static rt_mq_t mq;
static rt_sem_t sem;
static float data[ARRAY_SIZE];

static void producer_thread_entry(void *parameter)
{
    float temp_data;
    for (int i = 0; i < 50; i++)
    {
        // 采集數(shù)據(jù)
        temp_data = ...;  // 采集數(shù)據(jù)的邏輯
        // 發(fā)送數(shù)據(jù)到消息隊(duì)列
        rt_mq_send(mq, &temp_data, sizeof(float));
        // 判斷是否采集了50次數(shù)據(jù)
        if (i == 49)
        {
            // 發(fā)送信號量給消費(fèi)者線程
            rt_sem_release(sem);
        }
        rt_thread_mdelay(1000);  // 延時(shí) 1 秒
    }
}
static void consumer_thread_entry(void *parameter)
{
    float received_data[ARRAY_SIZE];
    while (1)
    {
        // 等待信號量
        rt_sem_take(sem, RT_WAITING_FOREVER);
        // 接收數(shù)據(jù)從消息隊(duì)列
        for (int i = 0; i < ARRAY_SIZE; i++)
        {
            rt_mq_recv(mq, &received_data[ i], sizeof(float), RT_WAITING_FOREVER);
        }
        // 處理接收到的數(shù)據(jù)
        for (int i = 0; i < ARRAY_SIZE; i++)
        {
            // 處理 received_data[ i]
        }
    }
}
int main(void)
{
    mq = rt_mq_create("mq", sizeof(float), 50, RT_IPC_FLAG_FIFO);
    if (mq == RT_NULL)
    {
        rt_kprintf("Failed to create message queue\n");
        return -1;
    }

    sem = rt_sem_create("sem", 0, RT_IPC_FLAG_FIFO);
    if (sem == RT_NULL)
    {
        rt_kprintf("Failed to create semaphore\n");
        return -1;
    }
    rt_thread_t tid1 = rt_thread_create("producer", producer_thread_entry, RT_NULL, 1024, 10, 10);
    if (tid1 != RT_NULL)
    {
        rt_thread_startup(tid1);
    }
    rt_thread_t tid2 = rt_thread_create("consumer", consumer_thread_entry, RT_NULL, 1024, 20, 10);
    if (tid2 != RT_NULL)
    {
        rt_thread_startup(tid2);
    }
    while (1)
    {
        rt_thread_mdelay(1000);
    }
}
[ i][ i]
這個時(shí)候,創(chuàng)建消息隊(duì)列的時(shí)候,就是
mq = rt_mq_create("mq", sizeof(float),//每個消息的最大字節(jié)數(shù)
                                     , 50                //消息隊(duì)列中最多有多少消息
                                     ,RT_IPC_FLAG_FIFO);

當(dāng)我們將50個數(shù)據(jù)封裝程一個數(shù)組,然后將數(shù)組一次性通過消息隊(duì)列發(fā)送給線程2的時(shí)候
例如 定義數(shù)組 float   value_data[50]; 存放數(shù)據(jù)
這個時(shí)候,創(chuàng)建的消息隊(duì)列應(yīng)該是
mq = rt_mq_create("mq", sizeof(float)*50,//每個消息(這里的每個消息就是指的這個數(shù)組是一個消息)的最大字節(jié)數(shù)
                                     , 3              //消息隊(duì)列中最多有多少消息,(隨便設(shè)定,實(shí)測,不易過大)
                                     ,RT_IPC_FLAG_FIFO);

這個特別注意,在創(chuàng)建消息隊(duì)列,傳輸數(shù)組的時(shí)候,數(shù)組作為消息發(fā)送的時(shí)候,單個消息的大小=sizeof(數(shù)組的存儲類型float)*數(shù)組深度

評分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏1 分享淘帖 頂 踩
回復(fù)

使用道具 舉報(bào)

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表