嵌入式系统C语言编程常用设计模式---参数表驱动设计
参数表驱动设计是一种软件开发和系统设计中常用的方法,它通过参数表来控制程序的行为和流程,提高系统的灵活性、可维护性和可扩展性。它将系统的行为逻辑与具体参数分离,通过表格形式集中管理配置信息。这种模式在嵌入式系统、工业控制和自动化领域尤为常见,特别适合处理复杂的报警 / 保护系统。
基本概念
参数表驱动设计是指将程序中一些可变的、需要根据不同情况进行调整的部分抽象出来,存储在一个或多个参数表中。程序在运行时,根据这些参数表中的信息来决定执行的逻辑、调用的模块或使用的算法等。参数表可以是数据库表、配置文件、内存数据结构等形式。
工作原理
- 参数存储:将程序中需要动态配置的信息集中存储在参数表中。这些信息可以包括业务规则、算法参数、界面显示格式、权限设置等。
- 参数读取:程序在运行时,根据需要从参数表中读取相应的参数。读取的方式可以是直接访问数据库、解析配置文件或访问内存数据结构。
- 逻辑控制:根据读取到的参数,程序动态地调整自身的行为。例如,根据不同的业务规则执行不同的计算逻辑,或者根据不同的界面显示格式生成不同的用户界面。
实际项目中应用参数表驱动设计
在实际项目中应用参数表驱动设计需要系统性的规划和实现。以下是一个完整的实施指南,结合具体案例说明:
1. 需求分析与场景评估
1.1 适用场景判断
参数表驱动设计适合以下场景:
- 系统包含大量可配置参数(如阈值、延时、使能位)
- 参数需要根据不同应用场景灵活调整
- 存在相似的处理逻辑但参数不同(如多级报警)
- 需要快速迭代或定制化开发
1.2 案例:电池管理系统(BMS)
- 参数类型:电压阈值、温度限制、电流限制、时间延时
- 处理逻辑:比较当前值与阈值,触发对应级别的报警或保护动作
- 变化需求:不同电池类型需要不同参数配置
2. 数据结构设计
2.1 定义枚举类型
typedef enum {// 电压类报警Alarm_CellVoltMax,Alarm_CellVoltMin,Alarm_TotalVoltMax,// 温度类报警Alarm_ChgTempHigh,Alarm_ChgTempLow,// ... 其他报警类型Alarm_Item_Num_Max // 枚举总数
} ET_AlarmVariety;
2.2 设计参数结构体
typedef struct {INT8U op; // 比较操作符 (GT, LT, EQ)INT8U proType; // 处理类型INT16U *pvar; // 监测变量指针INT16U threshold[3]; // 三级阈值INT16U time[3]; // 触发时间INT16U recovery[3]; // 恢复阈值INT16U recoveryTime[3]; // 恢复时间INT8U *statusBits; // 状态位指针
} ST_alarm_opt_t;
3. 参数表初始化
3.1 集中式初始化
static const ST_alarm_opt_t s_Alarm_opt_Init[Alarm_Item_Num_Max] = {[Alarm_CellVoltMax] = {.op = OP_GT,.proType = PROTECT_SHUTDOWN,.pvar = &g_cellVoltageMax,.threshold = {3600, 3800, 4000}, // 3.6V, 3.8V, 4.0V.time = {10, 5, 1}, // 触发时间(周期).recovery = {3500, 3700, 3900}, // 恢复阈值.recoveryTime = {20, 10, 5}, // 恢复时间.statusBits = &g_alarmStatus[Alarm_CellVoltMax]},// ... 其他报警配置
};
3.2 使用宏提高可读性
#define VOLT_TO_MV(volt) ((INT16U)((volt) * 1000))[Alarm_CellVoltMax] = {.threshold = {VOLT_TO_MV(3.6), VOLT_TO_MV(3.8), VOLT_TO_MV(4.0)},// ...
},
4. 核心访问接口实现
4.1 参数获取函数
/*** 获取指定报警类型的配置参数* @param alarmType 报警类型枚举值* @return 配置参数指针,失败返回NULL*/
ST_alarm_opt_t* GetAlarmConfig(ET_AlarmVariety alarmType) {if (alarmType >= Alarm_Item_Num_Max) {return NULL;}return &s_alarm_opt[alarmType];
}
4.2 通用报警检查函数
/*** 检查指定报警是否触发* @param alarmType 报警类型* @return 0-未触发,1-一级报警,2-二级报警,3-三级报警*/
INT8U CheckAlarmStatus(ET_AlarmVariety alarmType) {ST_alarm_opt_t* config = GetAlarmConfig(alarmType);if (!config || !config->pvar) return 0;INT16U currentValue = *(config->pvar);for (INT8U level = 3; level > 0; level--) {if (currentValue >= config->threshold[level-1]) {// 检查触发时间if (++config->triggerTime[level-1] >= config->time[level-1]) {return level;}} else {config->triggerTime[level-1] = 0; // 低于阈值,清零计时器}}return 0;
}
5. 集成与测试
5.1 初始化流程
void SystemInit(void) {// 1. 硬件初始化HardwareInit();// 2. 参数表初始化(从默认值或Flash加载)memcpy(s_alarm_opt, s_Alarm_opt_Init, sizeof(s_alarm_opt));LoadParametersFromFlash(); // 从非易失性存储加载自定义参数// 3. 参数验证if (!ValidateParameters()) {ResetToDefaultParameters();}// 4. 启动周期性任务StartPeriodicTasks();
}
5.2 周期性检查任务
void PeriodicTask_100ms(void) {// 检查所有报警for (ET_AlarmVariety alarm = 0; alarm < Alarm_Item_Num_Max; alarm++) {INT8U status = CheckAlarmStatus(alarm);if (status > 0) {HandleAlarm(alarm, status);}}
}
总结
参数表驱动设计的实施需要从需求分析、数据结构设计、接口实现到测试维护的全流程规划。通过合理分层、模块化设计和自动化工具,可以显著提高系统的灵活性、可维护性和开发效率。在资源受限的嵌入式系统中,这种设计模式尤其能发挥出最大优势。
设计模式解析
1. 核心结构关系
- 枚举类型
ET_AlarmVariety
:定义所有报警类型(如过压、过温),作为参数表的索引。 - 结构体
ST_alarm_opt_t
:封装每种报警的完整配置(阈值、延时、使能位等)。 - 静态常量数组
s_Alarm_opt_Init
:将枚举值与结构体实例一一映射,形成参数表。 - 宏定义:提供具体的数值配置(如 3.6V 阈值),增强可读性和可维护性。
2. 工作原理
通过枚举值作为数组索引,直接访问对应的配置参数:
// 获取"电池过压"的一级阈值
uint16_t threshold = s_alarm_opt[Alarm_CellVoltMax].para1;
系统运行时,通过查表方式动态获取配置,避免硬编码,提高灵活性。
设计优点
1. 模块化与可扩展性
- 分离逻辑与数据:报警检测逻辑(如比较、计时)与具体参数(如阈值、延时)分离。新增报警类型时,只需扩展枚举和参数表,无需修改核心逻辑。
- 统一管理:所有报警参数集中在一个数组中,便于维护和版本控制。
2. 代码复用与精简
- 通用处理函数:通过参数表驱动,可设计通用的报警检测函数:
void CheckAlarm(ET_AlarmVariety alarmType) {ST_alarm_opt_t* config = &s_alarm_opt[alarmType];uint16_t currentValue = *(uint16_t*)config->pvar;if (currentValue >= config->para1) { /* 触发一级报警 */ }
}
- 减少重复代码:避免为每种报警类型编写独立的检测逻辑。
3. 配置灵活性
- 参数动态调整:可通过修改宏定义或参数表初始化值,快速调整系统行为,无需重新编译核心代码。
- 运行时参数修改:若将参数表改为非 const,可在运行时动态调整报警阈值(如根据电池状态自适应调整)。
4. 可读性与可维护性
- 语义化配置:通过宏定义和枚举名,参数含义清晰(如
MAXCELLVOLTH_LEVELL2_SET
表示二级过压阈值)。 - 分层设计:从枚举→结构体→宏定义,形成清晰的层次结构,降低理解成本。
5. 资源优化
- 内存效率:参数表存储在 ROM(常量区),不占用 RAM 空间,适合资源受限的嵌入式系统。
- 代码体积:减少冗余逻辑,降低固件体积。
典型应用场景
- 电池管理系统(BMS):监测电压、温度、电流等参数,实现多级保护。
- 工业控制系统:对设备状态进行实时监控,触发不同级别的报警或保护动作。
- 汽车电子:如发动机管理系统、安全气囊触发逻辑。
- 智能家居:环境参数监测(如烟雾、温度)与报警联动。
对比其他设计模式
模式 | 适用场景 | 缺点 |
---|---|---|
硬编码参数 | 简单系统,参数固定 | 扩展性差,维护成本高 |
面向对象(策略模式) | 复杂行为变化 | 资源开销大,嵌入式系统慎用 |
参数表驱动 | 参数配置复杂,逻辑相对固定 | 需要精心设计数据结构 |
总结
参数表驱动设计通过数据抽象和集中管理,实现了系统的可扩展性、可维护性和灵活性,尤其适合嵌入式系统中的多参数监控与保护场景。这种设计将配置与逻辑解耦,使系统能够在不修改核心代码的情况下适应不同的应用需求,是工业控制领域的经典实践。
+-------+ +--------+
| 电压值 | ≥ 3.6V持续30个周期 → | 二级报警 |
+-------+ +--------+
| |
≤ 3.3V持续50个周期 |
↓ ↓
+-------+ +--------+
| 正常状态 | ←------------ | 报警恢复 |
+-------+ +--------+
相关文章:
嵌入式系统C语言编程常用设计模式---参数表驱动设计
参数表驱动设计是一种软件开发和系统设计中常用的方法,它通过参数表来控制程序的行为和流程,提高系统的灵活性、可维护性和可扩展性。它将系统的行为逻辑与具体参数分离,通过表格形式集中管理配置信息。这种模式在嵌入式系统、工业控制和自动…...

OpenCV CUDA模块图像过滤------创建一个行方向的一维积分(Sum)滤波器函数createRowSumFilter()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 cv::cuda::createRowSumFilter 是 OpenCV CUDA 模块中的一个函数,用于创建一个行方向的一维积分(Sum)滤波器。…...

Frequent values/gcd区间
Frequent values 思路: 这题它的数据是递增的,ST表,它的最多的个数只会在在两个区间本身就是最多的或中间地方产生,所以我用map数组储存每个值的左右临界点,在ST表时比较多一个比较中间值的个数就Ok了。 #define _…...

08SpringBoot高级--自动化配置
目录 Spring Boot Starter 依赖管理解释 一、核心概念 二、工作原理 依赖传递: 自动配置: 版本管理: 三、核心流程 四、常用 Starter 示例 五、自定义 Starter 步骤 创建配置类: 配置属性: 注册自动配置&a…...

Deep Evidential Regression
摘要 翻译: 确定性神经网络(NNs)正日益部署在安全关键领域,其中校准良好、鲁棒且高效的不确定性度量至关重要。本文提出一种新颖方法,用于训练非贝叶斯神经网络以同时估计连续目标值及其关联证据,从而学习…...

「Python教案」循环语句的使用
课程目标 1.知识目标 能使用for循环和while循环设计程序。能使用循环控制语句,break、continue、else设计程序。能使用循环实际问题。 2.能力目标 能根据需求合适的选择循环结构。能对嵌套循环代码进行调试和优化。能利用循环语句设计&am…...

linux快速入门-VMware安装linux,配置静态ip,使用服务器连接工具连接,快照和克隆以及修改相关配置信息
安装VMWare 省略,自己检索 安装操作系统-linux 注意:需要修改的我会给出标题,不要修改的直接点击下一步就可以 选择自定义配置 选择稍后安装操作系统 选择合适的内存 选择NAT模式 仅主机模式 虚拟机只能和主机通信,不能上网…...
用户配置文件(Profile)
2.4.5 用户配置文件(Profile) 用户配置文件由以下组件构成: 一个运营商安全域(MNO-SD) 辅助安全域(SSD)和CASD Applets 应用程序(如NFC应用) 网络接入应用ÿ…...
ubuntu 制作 ssl 证书
安装 openssl sudo apt install openssl 生成 SSL 证书 # 生成私钥 (Private Key) openssl genrsa -out private.key 2048 在当前目录生成 private.key # 生成证书签名请求 (CSR - Certificate Signing Request) openssl req -new -key private.key -out certificate.csr -…...
Vue组件技术全解析大纲
目录 01-全局组件 02-局部组件 03-组件属性 04-组件事件 05-组件插槽 06-生命周期 07-样式隔离 08-组件测试 09-组件发布 10-组件使用 开发优先级矩阵 01-全局组件 // 全局注册示例 Vue.component(global-button, {template: <button :style"btnStyle"…...

轻量化开源方案——浅析PdfPatcher实际应用
PDF处理在实际工作中十分重要,今天浅析PdfPatcher在PDF处理中的实际应用。 核心功能实测 批量处理能力 支持修改文档属性/页码编号/页面链接 一键清除复制/打印限制(实测WPS加密文档可解锁) 自动清理隐藏冗余数据(经测试可平均…...

Ansible常用Ad-Hoc 命令
1.配置sshpass yum install sshpass -y ssh-keygen -t dsa -f ~/.ssh/id_dsa -P "" # ssh-keygen密钥生成工具 -t密钥类型为dsa -f指定生成的密钥文件的路径。 -P:指定私钥的密码。 for i in seq 128 130; do sshpass -p123456 ssh-copy-id -i ~/.s…...

[论文阅读]Pandora: Jailbreak GPTs by Retrieval Augmented Generation Poisoning
Pandora: Jailbreak GPTs by Retrieval Augmented Generation Poisoning [2402.08416] Pandora: Jailbreak GPTs by Retrieval Augmented Generation Poisoning 间接越狱攻击 GPT的RAG增强过程分四个阶段:❶GPT首先组织不同的用户上传的文档类型(PDF、…...
鸿蒙OSUniApp 制作个性化的评分星级组件#三方框架 #Uniapp
UniApp 制作个性化的评分星级组件 在移动应用开发中,评分星级组件(Rating Star)是用户交互和反馈的重要工具,广泛应用于电商、外卖、内容社区等场景。一个美观、易用、可定制的评分组件,不仅能提升用户体验࿰…...

云效流水线Flow使用记录
概述 最近在频繁使用阿里云云效的几款产品,如流水线。之前写过一篇,参考云效流水线缓存问题。 这篇文章来记录更多问题。 环境变量 不管是云效流水线Flow还是应用交付AppStack(基于流水线,后文不再赘述)࿰…...

OpenCV CUDA模块图像处理------颜色空间处理之颜色空间转换函数cvtColor()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 该函数用于在 GPU 上进行颜色空间转换,支持多种常见的颜色空间转换操作。 函数原型 void cv::cuda::cvtColor (InputArray src…...

科技初创企业创新推动商业未来
在这个因变革而蓬勃发展的世界里,科技初创企业已成为各行业创新、颠覆与转型的驱动力。这些雄心勃勃的企业正在重塑商业格局,挑战既定规范,并不断突破可能性的边界。本文将深入探索科技初创企业的精彩领域,探讨它们如何通过创新塑…...
人工智能文科能学吗?
文科生也可以学习人工智能(AI),尽管这一领域传统上与数学和计算机科学联系紧密。然而,随着跨学科研究的发展,越来越多的人认识到文科背景在AI领域的价值。以下是一些文科生在学习AI时可以考虑的优势和需要克服的挑战&a…...
Ntfs!NtfsReadBootSector函数分析之nt!CcGetVacbMiss中得到一个nt!_VACB结构
第一部分: 1: kd> g Breakpoint 3 hit nt!CcGetVacbMiss: 80a1a19e 6a30 push 30h 1: kd> kc # 00 nt!CcGetVacbMiss 01 nt!CcGetVirtualAddress 02 nt!CcMapData 03 Ntfs!NtfsMapStream 04 Ntfs!NtfsReadBootSector Ntfs…...

猿大师办公助手WebOffice用二进制数据流在Web前端打开Office文档
猿大师办公助手作为第三代WebOffice方案,猿大师办公助手把本地原生Office无缝嵌入网页环境中实现在线编辑Office功能,提供了完全与本机Office一致(排版、打印等)的操作体验,保留100%原生功能(VBA宏、复杂公…...

etcd:高可用,分布式的key-value存储系统
引言 etcd是基于go语言开发的一款kv存储引擎,基于raft一致性算法实现的一种存储 一.etcd的底层原理 1.etcd的特点 高可用性与一致性:etcd 使用 Raft 算法保证集群中数据的强一致性,即使在节点故障的情况下也能保持数据完整性。 分布式存储&a…...

AI in Game,大模型能力与实时音视频技术融合,交出AI应用新答卷
随着AI的技术进步和工具普及,尤其是在这两年的跃进之后,AI在游戏行业内的应用已经逐步由理念设想推向落地实践。从蔡浩宇披露的AI新游《Whispers From The Star》到GDC上各大厂家呈现的游戏AI新亮点,我们看到了更多AI与游戏的结合方式&#x…...

欢乐熊大话蓝牙知识11:如何打造一个低功耗蓝牙温湿度传感器?
🧊 如何打造一个低功耗蓝牙温湿度传感器? 用电像抠门老头,通信像特工密谈。 🌡️ 引子:为什么你需要一个低功耗 BLE 传感器? 你是不是有过这种需求: 想在办公室角落放个传感器看温湿度,却不想拉电源线?想给智能养宠箱加个环境感知模块,但不能三天一换电池?想造个…...

Linux 安装 Remmina
欢迎关注公号:每日早参,第一时间获取AI资讯! 为什么安装Remmina, 因为Mobaxterm免费版本有窗口限制。 Remmina 是一款功能强大的开源远程桌面客户端,适用于 Linux 和其他类 Unix 系统,也支持 Windows 平台。 安装指南…...

什么是HTTP HTTP 和 HTTPS 的区别
HTTP协议定义 超文本传输协议(HyperText Transfer Protocol, HTTP)是一种应用层协议,主要用于客户端与服务器之间的数据交换。它基于请求-响应模型运行,在每次会话中由客户端发起请求,服务器返回相应的内容。 HTTP 是…...
cos和dmz学习
COS(Capability Open Service) 组件主要为系统提供能力开放的入口和控制。系统中需要对外进行能力开放的组件将RESTful的API接口注册到COS组件中,第三方系统就可以通过调用API来获取组件提供的能力。应用场景:当你想调用的外部系统接口不支持外网访问时&…...
上升沿计数 stm32 中断
在STM32上利用中断实现上升沿计数,可以按照以下步骤进行,这里以STM32F1系列为例,使用HAL库进行代码编写: 1. STM32CubeMX配置 打开STM32CubeMX并创建一个新工程,选择对应的STM32微控制器型号(如STM32F103C8T6)。在Pinout & Configuration选项卡中,找到用于检测上升…...
Java 各版本核心新特性的详细说明
一、Java 8(2014)—— 函数式编程的里程碑 1. Lambda 表达式 作用:简化匿名内部类,支持函数式编程。示例:// 传统匿名内部类 Runnable r1 new Runnable() {Overridepublic void run() {System.out.println("He…...
Nginx 性能优化全解析:从进程到安全的深度实践
一、进程优化:释放硬件性能潜力 Nginx 通过多工作进程处理请求,合理配置进程参数能充分利用 CPU 资源,避免资源浪费。 1.1 worker_processes 参数详解 worker_processes用于设置 Nginx 工作进程的数量,它直接影响 Nginx 对 CP…...

Pycharm and Flask 的学习心得(10)重定向
一 定义: 服务器告诉浏览器:你现在访问的这个页面,请改去另一个地址访问。 浏览器接收到这个“指令”后,会 自动跳转到另一个网页。 二 如何写: 方法一:重定向到网址 方法二:重定向到自己的…...