嵌入式軟體架構設計-消息交互

来源:https://www.cnblogs.com/const-zpc/archive/2022/07/05/16364418.html
-Advertisement-
Play Games

講述在單片機軟體開發過程中如何更好地實現各個模塊的數據交互,降低耦合 ...


1、前言

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

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


2、解決思路

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

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

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

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

看圖:

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

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

有以下幾個好處:

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

當然這種方式也有缺點:

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

避免方式:

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

3、示例代碼

事件通知模塊頭文件

#ifndef _NOTIFY_H_
#define _NOTIFY_H_


#include <stdint.h>


/**
  * @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 <string.h>

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

 

本文來自博客園,作者:大橙子瘋,轉載請註明原文鏈接:https://www.cnblogs.com/const-zpc/p/16364418.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文簡介 點贊 + 關註 + 收藏 = 學會了 ES6 推出的 const 可以定義常量。在 JS 中,常量是不可改變的。這個 “不可改變” 指的是常量存放的記憶體地址不變。 眾所周知,使用 const 定義的常量,如果是基礎類型的數據,值不能變。但如果是引用類型的數據(比如對象、數組等),是可以修改 ...
  • 本文簡介 點贊 + 關註 + 收藏 = 學會了 這是一次真實的 藍牙收發數據 的全過程講解。 本文使用 uni-app + Vue3 的方式進行開發,以手機app的方式運行(微信小程式同樣可行)。 uni-app 提供了 藍牙 和 低功耗藍牙 的 api ,和微信小程式提供的 api 是一樣的,所以 ...
  • 黑夜模式 作為一個前端學習者,自然懂得黑夜模式的重要性,可惜主題原生未提供,那就自己弄吧 個人博客作為效果參考:https://jieniyou.github.io/ 設置基礎樣式 參考其他優秀產品的黑夜模式,得出共性: 那就是黑夜模式的背景一般不會是純黑(#000);而是淡黑色,字體也不是純白(# ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 先從面向對象講起,本瓜認為:面向對象編程,它的最大能力就是:復用! 咱常說,面向對象三大特點,封裝、繼承、多態。 這三個特點,以“繼承”為核心。封裝成類,是為了繼承,繼承之後再各自發展(重寫),可理解為多態。所以,根本目的是為了繼承,即“ ...
  • 這18個網站是我在取經路上意外發現的,裡面包括 純CSS 實現的炫酷背景,還有專門製作背景圖的網站。 算是取經路上的大補之物~ 1. CSS3 Patterns Gallery 🎗️ 傳送門:『CSS3 Patterns Gallery』 如果你認識 Lea Verou 的話,大概率知道這個網站, ...
  • 最近,在 CodePen 上看到這樣一個非常有意思的效果: 這個效果的核心難點在於氣泡的一種特殊融合效果。 其源代碼在:CodePen Demo -- Goey footer,作者主要使用的是 SVG 濾鏡完成的該效果,感興趣的可以戳源碼看看。 其中,要想靈活運用 SVG 中的 feGaussian ...
  • 1、樣式 1.1 行內樣式 <h1 style="color:red;">行內樣式</h1> 1.2 內部樣式 CSS代碼寫在 <head> 的 <style> 標簽中 <style> h1{color: green; } </style> 1.3 外部樣式 <link rel="styleshee ...
  • 本文簡介 你負責點贊,我負責更新~ 這次要用純CSS做一個波點背景,先上圖看看效果。 我把這個效果寫在 body 上,如果你不喜歡這個配色也可以自己手動改改。 思路 我實現上圖的效果思路是,最先想到使用 background-image ,然後使用 radial-gradient 畫圓。再配合預設給 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...