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

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

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

嵌入式軟件架構(gòu)設(shè)計(jì)之消息交互

汽車電子技術(shù) ? 來源:大橙子瘋嵌入式 ? 作者: 大橙子瘋 ? 2023-02-15 15:44 ? 次閱讀

前言

在熟悉任務(wù)調(diào)度、程序分層和模塊化編程關(guān)于軟件架構(gòu)、分層和模塊設(shè)計(jì)后,除了函數(shù)調(diào)用設(shè)計(jì)中出現(xiàn)的情況外,還會(huì)遇到同層模塊之前如何進(jìn)行消息交互,通常是應(yīng)用層之間。

比如一個(gè)設(shè)備通過架構(gòu)設(shè)計(jì)包含人機(jī)交互應(yīng)用層模塊(一般會(huì)調(diào)用按鍵和顯示屏等功能驅(qū)動(dòng)模塊)和通信應(yīng)用層模塊(一般調(diào)用串口、CAN網(wǎng)絡(luò)ESP8266等功能驅(qū)動(dòng)模塊),兩個(gè)同層之間的模塊如果需要互傳數(shù)據(jù),一般都是調(diào)用各自頭文件提供的接口(模塊對(duì)外提供的接口盡量不要使用全局變量,防止其他模塊擅自修改),這樣就造成了耦合

設(shè)計(jì)思路

上述情況,也可以采用回調(diào)函數(shù)的實(shí)現(xiàn)方式進(jìn)行模塊解耦,但是需要引入新的內(nèi)容,即公共模塊Commoon層(包含第三方功能庫(kù))。圖片

公共模塊主要有各模塊都需要使用的類型定義、結(jié)構(gòu)體定義、通用函數(shù)或常用宏定義等(通常屬于基礎(chǔ)類的功能,不會(huì)受功能需求和不同平臺(tái)的影響)。

基于公共模塊,為了解決各模塊之前的數(shù)據(jù)交互,可以通過公共模塊實(shí)現(xiàn)基礎(chǔ)類的功能達(dá)到各應(yīng)用層模塊解耦的目的。

參考消息隊(duì)列的方式,可以實(shí)現(xiàn)一個(gè)生產(chǎn)者/消費(fèi)者的功能模塊(這種可以稱作觀察者模式,即存在觀察者和被觀察者),即某一模塊更新數(shù)據(jù)后,其他模塊可以第一時(shí)間得到通知更新(采用回調(diào)函數(shù)的方式實(shí)現(xiàn))

看圖:

圖片

Callback是一個(gè)指針數(shù)組變量,每個(gè)數(shù)組成員都是函數(shù)指針類型的變量,通過函數(shù)Notify_Attach拿到了應(yīng)用層代碼函數(shù)OnSaveParam(...)OnUpdateParam(...)的函數(shù)地址,之后人機(jī)交互模塊調(diào)用了Notify_EventNotify,從而調(diào)用Callback ,調(diào)用方式和直接調(diào)用 OnFunction(...) 存在些許差異,因?yàn)槭菙?shù)組,所有需要[]取函數(shù)地址,為了保證系統(tǒng)運(yùn)行安全,調(diào)用前要確保 Callback[i] 不為NULL,否則會(huì)引起程序異常。

從上述看,也許有人感覺這樣處理反而復(fù)雜了,直接調(diào)用不香嗎?(上述人機(jī)交互模塊屬于被觀察者,參數(shù)和其他模塊屬于觀察者)

有以下幾個(gè)好處:

  1. 避免各模塊相互調(diào)用,可完成解耦
  2. 即使 觀察者 模塊其中一個(gè)被移除,也不用修改 被觀察者 或者 其他觀察者 代碼,保證系統(tǒng)穩(wěn)定
  3. 新增一個(gè) 觀察者 模塊,也不需要修改 被觀察者 代碼,保證系統(tǒng)穩(wěn)定

當(dāng)然這種方式也有缺點(diǎn):

  1. 如果回調(diào)函數(shù)過多,或者某一個(gè) 觀察者 的回調(diào)函數(shù)執(zhí)行時(shí)間很長(zhǎng),肯定會(huì)影響到其他觀察者 模塊的通知時(shí)間,甚至影響 被觀察者 模塊的正常運(yùn)行
  2. 如果 觀察者 和 被觀察者 之間有循環(huán)依賴,就會(huì)導(dǎo)致他們循環(huán)調(diào)用,導(dǎo)致系統(tǒng)死機(jī)

避免方式:

  1. 回調(diào)函數(shù)中一定要保證執(zhí)行的時(shí)間短,不能有執(zhí)行時(shí)間長(zhǎng)的功能,甚至延時(shí)(一般回調(diào)中處理數(shù)據(jù)更新等執(zhí)行時(shí)間短的即可,數(shù)據(jù)更新后的需要花時(shí)間處理的可以在主循環(huán)執(zhí)行)
  2. 觀察者回調(diào)函數(shù)中盡量避免執(zhí)行其他觀察者的回調(diào)函數(shù),防止循環(huán)調(diào)用

示例代碼

下面簡(jiǎn)單實(shí)現(xiàn)人機(jī)交互模塊在某種情況下需要保存參數(shù),具體如何保存參數(shù)由參數(shù)模塊實(shí)現(xiàn),人機(jī)交互模塊通過事件通知模塊告知參數(shù)模塊需要保存數(shù)據(jù)。

初步來看,可能中間多了一個(gè),嫌實(shí)現(xiàn)麻煩,不如直接調(diào)用;但是從后期功能擴(kuò)展和解耦來看,這是很有必要的。

事件通知模塊

頭文件定義

#ifndef _NOTIFY_H_
#define _NOTIFY_H_


#include 


/**
  * @brief 應(yīng)用模塊ID枚舉定義
  *
  */
typedef enum
{
    NOTIFY_ID_HMI = 0,   // 人機(jī)交互模塊
    NOTIFY_ID_SYS_PARAM, // 參數(shù)管理模塊

    NOTIFY_ID_TOTAL
} NotifyId_e;

/**
  * @brief 事件類型枚舉定義
  *
  */
typedef enum
{
    NOTIFY_EVENT_PARAM_UPDATE,     // 參數(shù)更新事件, 對(duì)應(yīng)結(jié)構(gòu)體 PrramUpdateInfo_t

    NOTIFY_EVENT_TOTAL
} NotifyEvent_e;


typedef struct
{
    uint16_t addr;
    uint32_t param;
}PrramUpdateInfo_t;


typedef int (*EventNotifyCB)(NotifyId_e id, NotifyEvent_e eEvent, const void *pData, uint32_t length);


extern void Notify_Init(void);

extern int Notify_Attach(NotifyId_e id, NotifyEvent_e eEvent, EventNotifyCB pfnCallback);
extern int Notify_Detach(NotifyId_e id, NotifyEvent_e eEvent);
extern int Notify_EventNotify(NotifyId_e id, NotifyEvent_e eEvent, const void *pData, uint32_t length);

#endif /* _NOTIFY_H_ */

源文件實(shí)現(xiàn)

#include "notify.h"
#include 

static EventNotifyCB sg_pfnCallback[NOTIFY_ID_TOTAL][NOTIFY_EVENT_TOTAL];

/**
  * @brief      事件初始化
  *
  */
void Notify_Init(void)
{
    memset(sg_pfnCallback, 0, sizeof(sg_pfnCallback));
}

/**
  * @brief      添加事件監(jiān)聽通知
  *
  * @param[in]  id          應(yīng)用模塊ID
  * @param[in]  eEvent      事件
  * @param[in]  pfnCallback 回調(diào)函數(shù)
  * @return     0,成功; -1,失敗
  */
int Notify_Attach(NotifyId_e id, NotifyEvent_e eEvent, EventNotifyCB pfnCallback)
{
    if (id >= 0 && id < NOTIFY_ID_TOTAL && eEvent < NOTIFY_EVENT_TOTAL)
    {
        sg_pfnCallback[id][eEvent] = pfnCallback;
        return 0;
    }

    return -1;
}

/**
  * @brief      刪除事件監(jiān)聽通知
  *
  * @param[in]  id          應(yīng)用模塊ID
  * @param[in]  eEvent      事件
  * @return     0,成功; -1,失敗
  */
int Notify_Detach(NotifyId_e id, NotifyEvent_e eEvent)
{
    if (id >= 0 && id < NOTIFY_ID_TOTAL && eEvent < NOTIFY_EVENT_TOTAL)
    {
        sg_pfnCallback[id][eEvent] = 0;
        return 0;
    }

    return -1;
}

/**
  * @brief      事件通知
  *
  * @param[in]  id          應(yīng)用模塊ID
  * @param[in]  eEvent      事件類型
  * @param[in]  pData       消息內(nèi)容
  * @param[in]  length      消息長(zhǎng)度
  * @return     0,成功; -1,失敗
  */
int Notify_EventNotify(NotifyId_e id, NotifyEvent_e eEvent, const void *pData, uint32_t length)
{
    int i;

    if (eEvent < NOTIFY_EVENT_TOTAL)
    {
        for (i = 0; i < NOTIFY_ID_TOTAL; i++)
        {
            if (sg_pfnCallback[i][eEvent] != 0)
            {
                sg_pfnCallback[i][eEvent](id, eEvent, pData, length);
            }
        }

        return 0;
    }

    return -1;
}

參數(shù)應(yīng)用層模塊

示例通信,作為觀察者監(jiān)聽參數(shù)保存的消息。

#include "notify.h"

static int Param_OnNotifyProc(NotifyId_e id, NotifyEvent_e eEvent, const void *pData, uint32_t length);

void Param_Init(void)
{
    Notify_Attach(NOTIFY_ID_SYS_PARAM, NOTIFY_EVENT_PARAM_UPDATE, Param_OnNotifyProc);
}

// 事件回調(diào)處理
int Param_OnNotifyProc(NotifyId_e id, NotifyEvent_e eEvent, const void *pData, uint32_t length)
{
    switch (eEvent)
    {
    case NOTIFY_EVENT_PARAM_UPDATE:
        {
            PrramUpdateInfo_t *pInfo = (PrramUpdateInfo_t *)pData;
            SaveParam(pInfo->addr, pInfo->param);// 保存參數(shù)
        }
        break;
    default:
        break;
    }

    return 0;
}

人機(jī)交互應(yīng)用層模塊

示例通信,作為被觀察者通知/發(fā)送參數(shù)保存的消息。

#include "notify.h"

void Hmi_Init(void)
{

}

// 需要保存參數(shù)
int Hmi_SaveProc(void)
{
    ParamUpdateInfo_t info;

    info.addr = 5;
    info.param = 20;

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

    關(guān)注

    0

    文章

    64

    瀏覽量

    10253
  • 任務(wù)調(diào)度
    +關(guān)注

    關(guān)注

    0

    文章

    27

    瀏覽量

    9858
  • 模塊化編程
    +關(guān)注

    關(guān)注

    4

    文章

    17

    瀏覽量

    7708
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    嵌入式系統(tǒng)的軟件架構(gòu)設(shè)計(jì)!

    軟件架構(gòu)設(shè)計(jì)!2. 嵌入式環(huán)境下軟件設(shè)計(jì)的特點(diǎn)要談嵌入式軟件
    發(fā)表于 08-10 07:46

    為何要進(jìn)行嵌入式軟件架構(gòu)設(shè)計(jì)?如何設(shè)計(jì)?

    為何要進(jìn)行嵌入式軟件架構(gòu)設(shè)計(jì)?如何進(jìn)行嵌入式軟件架構(gòu)設(shè)計(jì)?
    發(fā)表于 11-01 06:31

    嵌入式軟件架構(gòu)設(shè)計(jì)的目的及思路

    【1】架構(gòu)設(shè)計(jì)的目的1.應(yīng)用的代碼邏輯清晰,且避免重復(fù)造輪子。2.方便軟件的移植。3.最大限度地復(fù)用。4.高內(nèi)聚低耦合。 【2】嵌入式架構(gòu)思路1.功能模塊化設(shè)計(jì)獲得需求------->
    發(fā)表于 11-08 06:41

    嵌入式UI架構(gòu)設(shè)計(jì)相關(guān)資料下載

    嵌入式UI架構(gòu)設(shè)計(jì)漫談
    發(fā)表于 11-08 07:49

    對(duì)嵌入式系統(tǒng)中的架構(gòu)設(shè)計(jì)的理解

    【閱讀這篇文章,你能了解到什么】1. 從事嵌入式開發(fā)12年的我,對(duì)架構(gòu)設(shè)計(jì)的理解;2. 對(duì)嵌入式系統(tǒng)中的架構(gòu)設(shè)計(jì)要刻意訓(xùn)練;3. 嵌入式系統(tǒng)
    發(fā)表于 11-08 08:23

    決定嵌入式系統(tǒng)軟件架構(gòu)的因素和架構(gòu)的影響

    嵌入式系統(tǒng)軟件架構(gòu)設(shè)計(jì)目錄1.前言42.決定架構(gòu)的因素和架構(gòu)的影響42.1.常見的誤解52.1.1.小型的系統(tǒng)不需要
    發(fā)表于 11-08 06:54

    嵌入式軟件架構(gòu)設(shè)計(jì)的資料大合集

    一、感慨近公司新招了一個(gè)做嵌入式軟件開發(fā)開發(fā)的童鞋,該童鞋是從上海的某一個(gè)上市公司出來的,因?yàn)槲覀冞@邊人手不夠,因此把他安排了去負(fù)責(zé)一個(gè)新產(chǎn)品的研發(fā),前期讓他負(fù)責(zé)加速度計(jì)、NB-IOT、舵機(jī)、外置
    發(fā)表于 11-09 07:50

    嵌入式軟件架構(gòu)設(shè)計(jì)資料分享

    作為程序員,我覺得如果要走的更遠(yuǎn)必須要成為工程師,畢竟年齡和資歷都擺在那里了。所以就讓我這個(gè)老程序員淺談一下嵌入式軟件架構(gòu)設(shè)計(jì)。我參考的也是一篇博文。原圖如下![在這里插入圖片描述](?x-oss-process=image/w
    發(fā)表于 12-24 07:09

    嵌入式軟件架構(gòu)設(shè)計(jì)

    嵌入式軟件架構(gòu)的設(shè)計(jì),幫助我們建立合理,有效的軟件架構(gòu)。
    發(fā)表于 11-09 17:34 ?19次下載

    探究嵌入式開發(fā)是否需要架構(gòu)設(shè)計(jì)?

    閱讀這篇文章,你能了解到什么 1. 從事嵌入式開發(fā)12年的我,對(duì)架構(gòu)設(shè)計(jì)的理解; 2. 對(duì)嵌入式系統(tǒng)中的架構(gòu)設(shè)計(jì)要刻意訓(xùn)練; 3. 嵌入式
    的頭像 發(fā)表于 04-05 09:49 ?3441次閱讀
    探究<b class='flag-5'>嵌入式</b>開發(fā)是否需要<b class='flag-5'>架構(gòu)設(shè)</b>計(jì)?

    嵌入式UI架構(gòu)設(shè)計(jì)漫談

    嵌入式UI架構(gòu)設(shè)計(jì)漫談
    發(fā)表于 11-03 17:36 ?15次下載
    <b class='flag-5'>嵌入式</b>UI<b class='flag-5'>架構(gòu)設(shè)</b>計(jì)漫談

    嵌入式開發(fā)需要架構(gòu)設(shè)計(jì)嗎?

    【閱讀這篇文章,你能了解到什么】1. 從事嵌入式開發(fā)12年的我,對(duì)架構(gòu)設(shè)計(jì)的理解;2. 對(duì)嵌入式系統(tǒng)中的架構(gòu)設(shè)計(jì)要刻意訓(xùn)練;3. 嵌入式系統(tǒng)
    發(fā)表于 11-03 18:06 ?15次下載
    <b class='flag-5'>嵌入式</b>開發(fā)需要<b class='flag-5'>架構(gòu)設(shè)</b>計(jì)嗎?

    嵌入式系統(tǒng)軟件架構(gòu)設(shè)計(jì)

    嵌入式系統(tǒng)軟件架構(gòu)設(shè)計(jì)目錄1.前言42.決定架構(gòu)的因素和架構(gòu)的影響42.1.常見的誤解52.1.1.小型的系統(tǒng)不需要
    發(fā)表于 11-03 18:21 ?30次下載
    <b class='flag-5'>嵌入式</b>系統(tǒng)<b class='flag-5'>軟件</b><b class='flag-5'>架構(gòu)設(shè)</b>計(jì)

    系統(tǒng)架構(gòu)設(shè)計(jì)筆記(59)—— 嵌入式系統(tǒng)的組成

    和支撐軟件是基礎(chǔ),應(yīng)用軟件則是最能體現(xiàn)整個(gè)嵌入式系統(tǒng)的特點(diǎn)和功能的部分。1 硬件架構(gòu)圖 1 是一個(gè)嵌入式系統(tǒng)的基本硬件
    發(fā)表于 11-04 11:06 ?13次下載
    系統(tǒng)<b class='flag-5'>架構(gòu)設(shè)</b>計(jì)筆記(59)—— <b class='flag-5'>嵌入式</b>系統(tǒng)的組成

    嵌入式系統(tǒng)的軟件架構(gòu)設(shè)計(jì)

    嵌入式軟件設(shè)計(jì)領(lǐng)域的一個(gè)分支,它自身的諸多特點(diǎn)決定了系統(tǒng)架構(gòu)師的選擇,同時(shí)它的一些問題又具有相當(dāng)?shù)耐ㄓ眯?,可以推廣到其他的領(lǐng)域。
    的頭像 發(fā)表于 03-12 11:06 ?4067次閱讀