C語言-訪問者模式詳解與實踐

C語言訪問者模式詳解與實踐 - 傳感器數據處理系統

1. 什么是訪問者模式?

在嵌入式系統中,我們經常需要對不同傳感器的數據進行多種處理,如數據校準、過濾、存儲等。訪問者模式允許我們在不修改傳感器代碼的情況下,添加新的數據處理方法。

2. 為什么需要訪問者模式?

  • 分離數據采集和處理邏輯
  • 便于添加新的數據處理方法
  • 避免修改現有傳感器代碼
  • 便于數據處理的復用
  • 結構清晰,易于維護

3. 實際應用場景

  • 多傳感器數據處理
  • 數據校準和濾波
  • 傳感器狀態監控
  • 數據記錄和分析
  • 故障診斷系統

4. 代碼實現

4.1 頭文件 (sensor_visitor.h)

#ifndef SENSOR_VISITOR_H
#define SENSOR_VISITOR_H#include <stdint.h>// 傳感器類型
typedef enum {TEMP_SENSOR,HUMIDITY_SENSOR
} SensorType;// 傳感器數據結構
typedef struct {uint16_t raw_data;    // 原始數據uint32_t timestamp;   // 時間戳SensorType type;      // 傳感器類型
} SensorData;// 溫度傳感器
typedef struct {SensorData data;float offset;         // 溫度補償值
} TempSensor;// 濕度傳感器
typedef struct {SensorData data;uint8_t threshold;    // 濕度閾值
} HumiditySensor;// 訪問者接口
typedef struct {void (*visit_temp)(void* self, TempSensor* sensor);void (*visit_humidity)(void* self, HumiditySensor* sensor);void* context;        // 訪問者上下文
} SensorVisitor;// 傳感器操作接口
void accept_visitor(void* sensor, SensorType type, SensorVisitor* visitor);// 創建傳感器
TempSensor* create_temp_sensor(uint16_t raw_data, float offset);
HumiditySensor* create_humidity_sensor(uint16_t raw_data, uint8_t threshold);// 創建訪問者
SensorVisitor* create_calibration_visitor(void);   // 校準訪問者
SensorVisitor* create_filter_visitor(void);        // 濾波訪問者
SensorVisitor* create_alarm_visitor(void);         // 報警訪問者// 銷毀函數
void destroy_sensor(void* sensor);
void destroy_visitor(SensorVisitor* visitor);#endif // SENSOR_VISITOR_H

4.2 實現文件 (sensor_visitor.c)

#include "sensor_visitor.h"
#include <stdio.h>
#include <stdlib.h>// 傳感器創建函數
TempSensor* create_temp_sensor(uint16_t raw_data, float offset) {TempSensor* sensor = (TempSensor*)malloc(sizeof(TempSensor));sensor->data.raw_data = raw_data;sensor->data.timestamp = 0;  // 實際應用中需要獲取真實時間戳sensor->data.type = TEMP_SENSOR;sensor->offset = offset;return sensor;
}HumiditySensor* create_humidity_sensor(uint16_t raw_data, uint8_t threshold) {HumiditySensor* sensor = (HumiditySensor*)malloc(sizeof(HumiditySensor));sensor->data.raw_data = raw_data;sensor->data.timestamp = 0;sensor->data.type = HUMIDITY_SENSOR;sensor->threshold = threshold;return sensor;
}// 校準訪問者實現
static void calibrate_temp(void* self, TempSensor* sensor) {float real_temp = sensor->data.raw_data * 0.1f + sensor->offset;printf("溫度校準: %.1f°C (原始值: %d, 偏移: %.1f)\n", real_temp, sensor->data.raw_data, sensor->offset);
}static void calibrate_humidity(void* self, HumiditySensor* sensor) {float real_humidity = sensor->data.raw_data * 0.1f;printf("濕度校準: %.1f%% (原始值: %d)\n", real_humidity, sensor->data.raw_data);
}// 濾波訪問者實現
static void filter_temp(void* self, TempSensor* sensor) {// 簡單的閾值濾波if (sensor->data.raw_data > 1000) {printf("溫度數據濾波: 數值異常\n");} else {printf("溫度數據濾波: 數值正常\n");}
}static void filter_humidity(void* self, HumiditySensor* sensor) {if (sensor->data.raw_data > 1000) {printf("濕度數據濾波: 數值異常\n");} else {printf("濕度數據濾波: 數值正常\n");}
}// 報警訪問者實現
static void alarm_temp(void* self, TempSensor* sensor) {float real_temp = sensor->data.raw_data * 0.1f + sensor->offset;if (real_temp > 50.0f) {printf("溫度報警: 溫度過高 (%.1f°C)\n", real_temp);}
}static void alarm_humidity(void* self, HumiditySensor* sensor) {if (sensor->data.raw_data > sensor->threshold) {printf("濕度報警: 超過閾值 %d\n", sensor->threshold);}
}// 創建訪問者
SensorVisitor* create_calibration_visitor(void) {SensorVisitor* visitor = (SensorVisitor*)malloc(sizeof(SensorVisitor));visitor->visit_temp = calibrate_temp;visitor->visit_humidity = calibrate_humidity;return visitor;
}SensorVisitor* create_filter_visitor(void) {SensorVisitor* visitor = (SensorVisitor*)malloc(sizeof(SensorVisitor));visitor->visit_temp = filter_temp;visitor->visit_humidity = filter_humidity;return visitor;
}SensorVisitor* create_alarm_visitor(void) {SensorVisitor* visitor = (SensorVisitor*)malloc(sizeof(SensorVisitor));visitor->visit_temp = alarm_temp;visitor->visit_humidity = alarm_humidity;return visitor;
}// accept函數實現
void accept_visitor(void* sensor, SensorType type, SensorVisitor* visitor) {switch (type) {case TEMP_SENSOR:visitor->visit_temp(visitor, (TempSensor*)sensor);break;case HUMIDITY_SENSOR:visitor->visit_humidity(visitor, (HumiditySensor*)sensor);break;}
}// 銷毀函數
void destroy_sensor(void* sensor) {free(sensor);
}void destroy_visitor(SensorVisitor* visitor) {free(visitor);
}

4.3 使用示例 (main.c)

#include "sensor_visitor.h"
#include <stdio.h>int main() {// 創建傳感器TempSensor* temp_sensor = create_temp_sensor(250, -2.5f);  // 25.0°C, -2.5偏移HumiditySensor* humidity_sensor = create_humidity_sensor(650, 800);  // 65.0%, 閾值80%// 創建訪問者SensorVisitor* calibration = create_calibration_visitor();SensorVisitor* filter = create_filter_visitor();SensorVisitor* alarm = create_alarm_visitor();// 處理溫度傳感器數據printf("=== 溫度傳感器處理 ===\n");accept_visitor(temp_sensor, TEMP_SENSOR, calibration);accept_visitor(temp_sensor, TEMP_SENSOR, filter);accept_visitor(temp_sensor, TEMP_SENSOR, alarm);// 處理濕度傳感器數據printf("\n=== 濕度傳感器處理 ===\n");accept_visitor(humidity_sensor, HUMIDITY_SENSOR, calibration);accept_visitor(humidity_sensor, HUMIDITY_SENSOR, filter);accept_visitor(humidity_sensor, HUMIDITY_SENSOR, alarm);// 清理資源destroy_sensor(temp_sensor);destroy_sensor(humidity_sensor);destroy_visitor(calibration);destroy_visitor(filter);destroy_visitor(alarm);return 0;
}

5. 代碼分析

5.1 關鍵設計點

  1. 傳感器數據結構清晰
  2. 處理邏輯分離
  3. 易于擴展新的處理方法
  4. 資源管理完善

5.2 實現特點

  1. 適合嵌入式系統
  2. 代碼簡潔高效
  3. 內存占用小
  4. 處理流程清晰

6. 編譯和運行

gcc -c sensor_visitor.c -o sensor_visitor.o
gcc -c main.c -o main.o
gcc sensor_visitor.o main.o -o sensor_demo

7. 注意事項

  1. 內存使用要謹慎
  2. 避免過度設計
  3. 考慮實時性要求
  4. 錯誤處理要完善

8. 改進建議

  1. 添加數據存儲功能
  2. 實現數據平滑處理
  3. 添加更多傳感器類型
  4. 支持批量數據處理

9. 總結

這個例子展示了訪問者模式在嵌入式系統中的實際應用。通過將傳感器數據采集和處理邏輯分離,我們可以靈活地添加新的數據處理方法,同時保持代碼的清晰和可維護性。

參考資料

  1. 《嵌入式系統設計》
  2. 《C語言程序設計》
  3. 《傳感器技術手冊》

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/74297.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/74297.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/74297.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

(UI自動化測試web端)第二篇:元素定位的方法_xpath路徑定位

1、第一種xpath路徑定位&#xff1a; 絕對路徑&#xff1a;表達式是以 /html開頭&#xff0c;元素的層級之間是以 / 分隔相同層級的元素可以使用下標&#xff0c;下標是從1開始的需要列出元素所經過的所有層級元素&#xff0c;工作當中一般不使用絕對路徑 例&#xff1a;/html/…

設置GeoJSONVectorTileLayer中的line填充圖片

設置GeoJSONVectorTileLayer中的line填充圖片 關鍵&#xff1a;linePatternFile const style [{filter: true,renderPlugin: {dataConfig: {type: "line",},type: "line",},symbol: {linePatternFile: "http://examples.maptalks.com/resources/pat…

electron框架(4.0)electron-builde和electron Forge的打包方式

----使用electron-builder打包&#xff08;需要魔法&#xff09; --安裝electron-builder: npm install electron-builder -D--package.json中進行相關配置&#xff1a; {"name": "video-tools","version": "1.0.0","main&quo…

讓 MGR 不從 Primary 的節點克隆數據?

問題 MGR 中&#xff0c;新節點在加入時&#xff0c;為了與組內其它節點的數據保持一致&#xff0c;它會首先經歷一個分布式恢復階段。在這個階段&#xff0c;新節點會隨機選擇組內一個節點&#xff08;Donor&#xff09;來同步差異數據。 在 MySQL 8.0.17 之前&#xff0c;同…

第三十二篇 深入解析Kimball維度建模:構建企業級數據倉庫的完整框架

目錄 一、維度建模設計原則深度剖析1.1 業務過程驅動設計1.2 星型模式VS雪花模式 二、維度建模五步法實戰&#xff08;附完整案例&#xff09;2.1 業務需求映射2.2 模型詳細設計2.3 緩慢變化維處理 三、高級建模技術解析3.1 漸變維度橋接表3.2 快照事實表設計 四、性能優化體系…

IntelliJ IDEA 中 Maven 的 `pom.xml` 變灰帶橫線?一文詳解解決方法

前言 在使用 IntelliJ IDEA 進行 Java 開發時&#xff0c;如果你發現項目的 pom.xml 文件突然變成灰色并帶有刪除線&#xff0c;這可能是 Maven 的配置或項目結構出現了問題。 一、問題現象與原因分析 現象描述 文件變灰&#xff1a;pom.xml 在項目資源管理器中顯示為灰色。…

緩存過期時間之邏輯過期

1. 物理不過期&#xff08;Physical Non-Expiration&#xff09; 定義&#xff1a;在Redis中不設置EXPIRE時間&#xff0c;緩存鍵永久存在&#xff08;除非主動刪除或內存淘汰&#xff09;。目的&#xff1a;徹底規避因緩存自動過期導致的擊穿&#xff08;單熱點失效&#xff…

基于WebAssembly的瀏覽器密碼套件

目錄 一、前言二、WebAssembly與瀏覽器密碼套件2.1 WebAssembly技術概述2.2 瀏覽器密碼套件的需求三、系統設計思路與架構3.1 核心模塊3.2 系統整體架構圖四、核心數學公式與算法證明4.1 AES-GCM加解密公式4.2 SHA-256哈希函數五、異步任務調度與GPU加速設計5.1 異步任務調度5.…

Qt的內存管理機制

在Qt中&#xff0c;顯式使用new創建的對象通常不需要顯式調用delete來釋放內存&#xff0c;這是因為Qt提供了一種基于對象樹(Object Tree)和父子關系(Parent-Child Relationship)的內存管理機制。這種機制可以自動管理對象的生命周期&#xff0c;確保在適當的時候釋放內存&…

數據結構之雙向鏈表-初始化鏈表-頭插法-遍歷鏈表-獲取尾部結點-尾插法-指定位置插入-刪除節點-釋放鏈表——完整代碼

數據結構之雙向鏈表-初始化鏈表-頭插法-遍歷鏈表-獲取尾部結點-尾插法-指定位置插入-刪除節點-釋放鏈表——完整代碼 #include <stdio.h> #include <stdlib.h>typedef int ElemType;typedef struct node{ElemType data;struct node *next, *prev; }Node;//初化鏈表…

【Linux網絡-五種IO模型與阻塞IO】

一、引入 網絡通信的本質就是進程間的通信&#xff0c;進程間通信的本質就是IO&#xff08;Input&#xff0c;Output&#xff09; I/O&#xff08;input/output&#xff09;也就是輸入和輸出&#xff0c;在馮諾依曼體系結構當中&#xff0c;將數據從輸入設備拷貝到內存就叫作…

算法-最大公約數

1、約數&#xff1a; 1.1 試除法求約數 原理&#xff1a;只需要遍歷最小的約數即可&#xff0c;較大的那個可以直接算出來。 import java.util.*; public class Main {static Scanner sc new Scanner(System.in);public static void main(String[] args) {int t sc.nextIn…

湖北楚大夫

品牌出海已成為眾多企業拓展業務、提升競爭力的關鍵戰略。楚大夫(chudafu.com)作為一家專注于品牌出海、海外網絡營銷推廣以及外貿獨立站搭建的公司&#xff0c;憑借其專業、高效、創新的服務模式&#xff0c;致力于成為中國企業走向國際市場的堅實后盾與得力伙伴。楚大夫通過綜…

Flutter 學習之旅 之 flutter 使用 connectivity_plus 進行網路狀態監聽(斷網/網絡恢復事件監聽)

Flutter 學習之旅 之 flutter 使用 connectivity_plus 進行網路狀態監聽&#xff08;斷網/網絡恢復事件監聽&#xff09; 目錄 Flutter 學習之旅 之 flutter 使用 connectivity_plus 進行網路狀態監聽&#xff08;斷網/網絡恢復事件監聽&#xff09; 一、簡單介紹 二、conne…

從零開始實現 C++ TinyWebServer 處理請求 HttpRequest類詳解

文章目錄 HTTP 請求報文HttpRequest 類實現 Init() 函數實現 ParseRequestLine() 函數實現 ParseHeader() 函數實現 ParsePath() 函數實現 ParseBody() 函數實現 ParsePost() 函數實現 ParseFromUrlEncoded() 函數實現 UserVerify() 函數實現 Parse() 函數HttpRequest 代碼Http…

systemd-networkd 的 *.network 配置文件詳解 筆記250323

systemd-networkd 的 *.network 配置文件詳解 筆記250323 查看官方文檔可以用 man systemd.network命令, 或訪問: https://www.freedesktop.org/software/systemd/man/latest/systemd.network.html 名稱 systemd.network — 網絡配置 概要 network.network 描述 一個純…

自定義mavlink 生成wireshark wlua插件錯誤(已解決)

進入正題 python3 -m pymavlink.tools.mavgen --langWLua --wire-protocol2.0 --outputoutput/develop message_definitions/v1.0/development.xml 編譯WLUA的時候遇到一些問題 1.ERROR:SCHEMASV:SCHEMAV_CVC_ENUMERATION_VALID 3765:0:ERROR:SCHEMASV:SCHEMAV_CVC_ENUMERAT…

計算機操作系統(四) 操作系統的結構與系統調用

計算機操作系統&#xff08;四&#xff09; 操作系統的結構與系統調用 前言一、操作系統的結構1.1 簡單結構1.2 模塊化結構1.3 分層化結構1.4 微內核結構1.5 外核結構 二、系統調用1.1 系統調用的基本概念1.2 系統調用的類型 總結&#xff08;核心概念速記&#xff09;&#xf…

深入解析 Spring IOC AOP:原理、源碼與實戰

深入解析 Spring IOC & AOP&#xff1a;原理、源碼與實戰 Spring 框架的核心在于 IOC&#xff08;控制反轉&#xff09; 和 AOP&#xff08;面向切面編程&#xff09;。今天&#xff0c;我們將深入剖析它們的原理&#xff0c;結合源碼解析&#xff0c;并通過 Java 代碼實戰…

LLM之RAG理論(十四)| RAG 最佳實踐

RAG 的過程很復雜&#xff0c;包含許多組成部分。我們如何確定現有的 RAG 方法及其最佳組合&#xff0c;以確定最佳 RAG 實踐&#xff1f; 論文 《Searching for Best Practices in Retrieval-Augmented Generation》給出了回答。 本文將從以下三方面進行介紹&#xff1a; 首先…