<acronym id="s8ci2"><small id="s8ci2"></small></acronym>
<rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
<acronym id="s8ci2"></acronym>
<acronym id="s8ci2"><center id="s8ci2"></center></acronym>
0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

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

3天內不再提示

嵌入式軟件架構設計之消息交互

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

前言

在熟悉任務調度、程序分層和模塊化編程關于軟件架構、分層和模塊設計后,除了函數調用設計中出現的情況外,還會遇到同層模塊之前如何進行消息交互,通常是應用層之間。

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

設計思路

上述情況,也可以采用回調函數的實現方式進行模塊解耦,但是需要引入新的內容,即公共模塊Commoon層(包含第三方功能庫)。圖片

公共模塊主要有各模塊都需要使用的類型定義、結構體定義、通用函數或常用宏定義等(通常屬于基礎類的功能,不會受功能需求和不同平臺的影響)。

基于公共模塊,為了解決各模塊之前的數據交互,可以通過公共模塊實現基礎類的功能達到各應用層模塊解耦的目的。

參考消息隊列的方式,可以實現一個生產者/消費者的功能模塊(這種可以稱作觀察者模式,即存在觀察者和被觀察者),即某一模塊更新數據后,其他模塊可以第一時間得到通知更新(采用回調函數的方式實現)

看圖:

圖片

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

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

有以下幾個好處:

  1. 避免各模塊相互調用,可完成解耦
  2. 即使 觀察者 模塊其中一個被移除,也不用修改 被觀察者 或者 其他觀察者 代碼,保證系統穩定
  3. 新增一個 觀察者 模塊,也不需要修改 被觀察者 代碼,保證系統穩定

當然這種方式也有缺點:

  1. 如果回調函數過多,或者某一個 觀察者 的回調函數執行時間很長,肯定會影響到其他觀察者 模塊的通知時間,甚至影響 被觀察者 模塊的正常運行
  2. 如果 觀察者 和 被觀察者 之間有循環依賴,就會導致他們循環調用,導致系統死機

避免方式:

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

示例代碼

下面簡單實現人機交互模塊在某種情況下需要保存參數,具體如何保存參數由參數模塊實現,人機交互模塊通過事件通知模塊告知參數模塊需要保存數據。

初步來看,可能中間多了一個,嫌實現麻煩,不如直接調用;但是從后期功能擴展和解耦來看,這是很有必要的。

事件通知模塊

頭文件定義

#ifndef _NOTIFY_H_
#define _NOTIFY_H_


#include 


/**
  * @brief 應用模塊ID枚舉定義
  *
  */
typedef enum
{
    NOTIFY_ID_HMI = 0,   // 人機交互模塊
    NOTIFY_ID_SYS_PARAM, // 參數管理模塊

    NOTIFY_ID_TOTAL
} NotifyId_e;

/**
  * @brief 事件類型枚舉定義
  *
  */
typedef enum
{
    NOTIFY_EVENT_PARAM_UPDATE,     // 參數更新事件, 對應結構體 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_ */

源文件實現

#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      添加事件監聽通知
  *
  * @param[in]  id          應用模塊ID
  * @param[in]  eEvent      事件
  * @param[in]  pfnCallback 回調函數
  * @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      刪除事件監聽通知
  *
  * @param[in]  id          應用模塊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          應用模塊ID
  * @param[in]  eEvent      事件類型
  * @param[in]  pData       消息內容
  * @param[in]  length      消息長度
  * @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;
}

參數應用層模塊

示例通信,作為觀察者監聽參數保存的消息。

#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);
}

// 事件回調處理
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);// 保存參數
        }
        break;
    default:
        break;
    }

    return 0;
}

人機交互應用層模塊

示例通信,作為被觀察者通知/發送參數保存的消息。

#include "notify.h"

void Hmi_Init(void)
{

}

// 需要保存參數
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));
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 軟件架構
    +關注

    關注

    0

    文章

    61

    瀏覽量

    10232
  • 任務調度
    +關注

    關注

    0

    文章

    27

    瀏覽量

    9839
  • 模塊化編程
    +關注

    關注

    4

    文章

    17

    瀏覽量

    7697
收藏 人收藏

    評論

    相關推薦

    誠聘嵌入式軟件架構

    ,本科以上學歷;2、三年以上嵌入式軟件系統架構設計經驗,同時有通信行業軟件開發經驗;3、精通C/C++語言,精通數據結構;熟悉QT\linux\android
    發表于 03-01 10:20

    嵌入式系統的軟件架構設計!

    軟件架構設計!2. 嵌入式環境下軟件設計的特點要談嵌入式軟件
    發表于 08-10 07:46

    嵌入式Linux系統知識架構

    嵌入式Linux系統知識架構及層次嵌入式Linux系統構成及啟動略析嵌入式Linux三劍客uboot技術
    發表于 10-27 07:22

    嵌入式軟件架構設計的相關資料分享

    嵌入式軟件架構的設計大多數嵌入式程序員學習編程,都是從開發板的附帶例程開始。之后工作也會繼續參考那些例程,很多編程習慣、方式也會受之影響。其實開發板式的編程方式與工作中實際需求的并不完
    發表于 10-28 06:15

    為何要進行嵌入式軟件架構設計?如何設計?

    為何要進行嵌入式軟件架構設計?如何進行嵌入式軟件架構設計?
    發表于 11-01 06:31

    嵌入式軟件架構設計的目的及思路

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

    嵌入式UI架構設計相關資料下載

    嵌入式UI架構設計漫談
    發表于 11-08 07:49

    嵌入式系統中的架構設計的理解

    【閱讀這篇文章,你能了解到什么】1. 從事嵌入式開發12年的我,對架構設計的理解;2. 對嵌入式系統中的架構設計要刻意訓練;3. 嵌入式系統
    發表于 11-08 08:23

    決定嵌入式系統軟件架構的因素和架構的影響

    嵌入式系統軟件架構設計目錄1.前言42.決定架構的因素和架構的影響42.1.常見的誤解52.1.1.小型的系統不需要
    發表于 11-08 06:54

    嵌入式軟件架構設計的資料大合集

    一、感慨近公司新招了一個做嵌入式軟件開發開發的童鞋,該童鞋是從上海的某一個上市公司出來的,因為我們這邊人手不夠,因此把他安排了去負責一個新產品的研發,前期讓他負責加速度計、NB-IOT、舵機、外置
    發表于 11-09 07:50

    嵌入式軟件架構設計資料分享

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

    嵌入式軟件架構設

    嵌入式軟件架構的設計,幫助我們建立合理,有效的軟件架構。
    發表于 11-09 17:34 ?19次下載

    嵌入式UI架構設計漫談

    嵌入式UI架構設計漫談
    發表于 11-03 17:36 ?15次下載
    <b class='flag-5'>嵌入式</b>UI<b class='flag-5'>架構設</b>計漫談

    嵌入式開發需要架構設計嗎?

    【閱讀這篇文章,你能了解到什么】1. 從事嵌入式開發12年的我,對架構設計的理解;2. 對嵌入式系統中的架構設計要刻意訓練;3. 嵌入式系統
    發表于 11-03 18:06 ?15次下載
    <b class='flag-5'>嵌入式</b>開發需要<b class='flag-5'>架構設</b>計嗎?

    嵌入式系統軟件架構設

    嵌入式系統軟件架構設計目錄1.前言42.決定架構的因素和架構的影響42.1.常見的誤解52.1.1.小型的系統不需要
    發表于 11-03 18:21 ?29次下載
    <b class='flag-5'>嵌入式</b>系統<b class='flag-5'>軟件</b><b class='flag-5'>架構設</b>計
    亚洲欧美日韩精品久久_久久精品AⅤ无码中文_日本中文字幕有码在线播放_亚洲视频高清不卡在线观看
    <acronym id="s8ci2"><small id="s8ci2"></small></acronym>
    <rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
    <acronym id="s8ci2"></acronym>
    <acronym id="s8ci2"><center id="s8ci2"></center></acronym>