C语言从句柄到对象

C语言从句柄到对象 (一) —— 全局变量的噩梦与“多实例”的救赎

代码里的句柄(Handle) 到底是个什么东西?为什么大厂的代码库(SDK)里到处都是句柄?” 其实,“句柄” (Handle) 不仅仅是一个指针,它是 C 语言通向模块化和面向对象架构的第一把钥匙。 今天,我们不谈枯燥的语法,只谈一个最痛的实际问题:当产品需求从“控制 1 个电机”变成“控制 10 个电机”时,你的代码需要推倒重写吗?

一、 这种“菜鸟代码”,你一定写过

假设我们接到了一个任务:写一个电机驱动,控制转速和启停。 对于大多数初学者,或者在为了赶进度的场景下,代码通常是这样写出来的:

1.1 典型的“隐式单例”写法

motor.c (驱动文件) 我们把电机的状态定义为文件内部的全局变量(static),然后用函数直接操作它们。

// [数据区] 全局变量:隐式地只支持一个设备 static uint8_t g_current_speed = 0; static uint8_t g_is_running = 0; // [代码区] 操作函数 void Motor_Init(void) { HAL_GPIO_Init(GPIOA, GPIO_PIN_0, ...); // 硬件引脚写死在代码里 g_current_speed = 0; } void Motor_SetSpeed(uint8_t speed) { g_current_speed = speed; // ... 操作硬件寄存器 ... }

main.c (应用文件)

int main(void) { Motor_Init(); Motor_SetSpeed(50); // 设置速度 }

1.2 这种写法有问题吗?

坦白说,如果你的板子上这辈子只会有 1 个电机,这种写法完全没问题。 这种写法叫 “单例模式” (Singleton)。它的优点是简单、直观、调用方便,不需要传参。在嵌入式系统中,对于 全局唯一的资源(比如系统滴答定时器 SysTick、唯一的看门狗 WDT、调试日志 Log),这种写法甚至是推荐的。

但是,噩梦往往源于那句:“老板说,新产品要加功能。”

二、 复制粘贴的灾难

老板突然走过来说:“这一版硬件升级了,我们需要控制 4 个电机。”

这时候,如果你坚持上面的写法,你只能祭出“复制粘贴大法”:

// 这种写法被称为“维护者的噩梦” // --- 电机 1 --- static uint8_t g_speed_1 = 0; void Motor1_SetSpeed(uint8_t speed) { ... } // --- 电机 2 --- static uint8_t g_speed_2 = 0; void Motor2_SetSpeed(uint8_t speed) { ... } // --- 电机 3 ... (代码量膨胀 4 倍)

后果是显而易见的: 代码冗余:同样的逻辑重复了 4 遍,Flash 空间被浪费。 维护困难:如果发现电机启动逻辑有个 Bug,你得去改 4 个地方。只要漏改一个,Bug 就依然存在。 无法扩展:如果明天老板要 10 个电机呢? 这就是 “面向过程” 编程在面对 “多实例” 需求时的死穴。

三、 句柄的诞生:将“数据”与“逻辑”剥离

为了解决这个问题,我们需要转换思维。 仔细观察上面的代码,你会发现:

变的(数据):每个电机的引脚、当前速度、运行状态,是各自独立的。

不变的(逻辑):设置 PWM、计算转速的算法(代码指令),对所有电机都是一样的。 我们为什么不能 只写一份代码,让它去操作 多份数据 呢?

3.1 第一步:定义对象 (The Context)

我们把原本散落在全局变量里的东西,全部“打包”进一个结构体。这个结构体,在架构术语里叫 Context(上下文),或者是 Instance(实例)。

// motor_driver.h typedef struct { // 静态属性 (配置) GPIO_TypeDef *gpio_port; uint16_t gpio_pin; // 动态状态 (变量) uint8_t current_speed; uint8_t is_running; } Motor_t;

3.2 第二步:引入句柄 (The Handle)

现在,我们的函数不再操作死板的全局变量,而是操作作为参数传进来的指针。 这个指针,就是我们俗称的 “句柄”。

// motor_driver.h // 定义句柄类型:也就是指向 Motor_t 的指针 typedef Motor_t* Motor_Handle; // 接口函数的第一个参数,永远是句柄! // 翻译过来就是:“你要让我操作哪一个电机?” void Motor_Init(Motor_Handle h, GPIO_TypeDef *port, uint16_t pin); void Motor_SetSpeed(Motor_Handle h, uint8_t speed);

3.3 第三步:实现逻辑 (Implementation)

在 .c 文件里,我们通过句柄(指针)来访问具体的数据。

// motor_driver.c void Motor_Init(Motor_Handle h, GPIO_TypeDef *port, uint16_t pin) { // 【核心】把配置信息存入句柄指向的内存区域 h->gpio_port = port; h->gpio_pin = pin; h->current_speed = 0; // 使用句柄里的信息初始化硬件 HAL_GPIO_Init(h->gpio_port, h->gpio_pin, ...); } void Motor_SetSpeed(Motor_Handle h, uint8_t speed) { // 1. 修改句柄指向的状态 h->current_speed = speed; // 2. 操作句柄指向的硬件 Set_PWM_Hardware(h->gpio_port, h->gpio_pin, speed); }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/1141644.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java Web 洗衣店订单管理系统系统源码-SpringBoot2+Vue3+MyBatis-Plus+MySQL8.0【含文档】

💡实话实说:用最专业的技术、最实惠的价格、最真诚的态度服务大家。无论最终合作与否,咱们都是朋友,能帮的地方我绝不含糊。买卖不成仁义在,这就是我的做人原则。摘要 随着互联网技术的快速发展,传统洗衣店…

RabbitMQ 的介绍与使用

一. 简介 1> 什么是MQ 消息队列(Message Queue,简称MQ),从字面意思上看,本质是个队列,FIFO先入先出,只不过队列中存放的内容是message而已。 其主要用途:不同进程Process/线程T…

RabbitMQ HAProxy 负载均衡

文章目录 前言当Java中指定的端口号绑定的rabbitmq服务挂掉了之后,我们的程序是否还能够成功访问到rabbitmq服务呢什么是 HAProxy 负载均衡HAProxy 安装修改HAProxy配置文件使用HAProxy结论 前言 前面我们学习了 rabbitmq 搭建集群,并且为了解决集群中…

RISC架构下实时操作系统移植:项目应用

RISC架构下实时操作系统移植:从原理到实战的深度实践在工业自动化、智能驾驶和边缘计算飞速发展的今天,嵌入式系统早已不再是“跑个循环”的简单设备。越来越多的应用要求毫秒级响应、任务间精确协同、资源高效调度——这些正是实时操作系统(…

STM32在Proteus 8 Professional中的仿真可行性深度剖析

STM32能在Proteus里“跑起来”吗?——一次不绕弯的仿真实战复盘最近带学生做课程设计,又碰上了那个老问题:“老师,我还没拿到开发板,能不能先用Proteus仿真一下STM32的代码?”这问题听着简单,但…

从零开始:使用Hadoop处理物联网数据的完整指南

从零开始:使用Hadoop处理物联网数据的完整指南关键词:Hadoop、物联网数据、数据处理、分布式计算、大数据摘要:本文旨在为读者提供一份从零基础开始,使用Hadoop处理物联网数据的完整指南。首先介绍了物联网数据处理的背景和使用Ha…

CAPL实现远程诊断请求自动响应:实战案例

用CAPL打造“会说话”的虚拟ECU:远程诊断自动响应实战全解析你有没有遇到过这样的场景?新项目刚启动,硬件还没影儿,测试团队却急着要验证诊断协议;或者产线检测卡在某个负响应逻辑上,真实ECU死活不肯配合复…

Betaflight在F4飞控板上的配置优化:全面讲解

Betaflight在F4飞控板上的配置优化:从底层机制到飞行手感的全面调校 你有没有过这样的体验?——刚组装好一台穿越机,装上高端电机、轻量化机架、碳纤螺旋桨,结果一飞起来却“软绵无力”,转弯拖泥带水,油门…

永磁同步电机无差拍预测控制加延时补偿:探索高效电机控制之路

永磁同步电机无差拍预测控制加延时补偿在电机控制领域,永磁同步电机(PMSM)凭借其高功率密度、高效率等优点,广泛应用于工业、交通等众多领域。而如何实现对PMSM的精准控制,一直是研究的热点。今天咱们就来聊聊永磁同步…

[内网流媒体] 零信任理念在内网工具中的落地

零信任的核心 零信任强调“永不信任,始终验证”。即便在内网,也假设网络不可信、设备不可信、用户可能被劫持。对实时画面工具,零信任的落地关乎访问控制、最小权限和持续验证。 落地原则 身份优先 所有访问都需身份验证(口令/Token/单点登录),不提供匿名入口。 最小权…

Kafka Connect详解:大数据ETL的得力助手

Kafka Connect详解:大数据ETL的得力助手 关键词:Kafka Connect、ETL、数据管道、连接器、分布式系统、数据集成、大数据 摘要:本文将深入探讨Kafka Connect的核心概念和工作原理,这个专为Apache Kafka设计的可扩展、可靠的数据集成…

vh6501测试busoff:硬件工程师实战案例解析

vh6501测试Bus-Off:硬件工程师的实战指南从一个真实问题说起某新能源车型在路试中偶发“整车通信中断”故障,仪表黑屏、动力降级。售后排查未发现硬件损坏,日志显示BMS模块突然停止发送报文,但其他节点并未崩溃。最终定位到&#…

模拟电子技术驱动的振荡器设计:从零实现教程

从零构建一个正弦波振荡器:模拟电路的艺术与实战 你有没有试过,只用几个电阻、电容和一块运放,让电路“自己”发出稳定的正弦波?没有单片机、没有代码、也没有复杂的数字逻辑——一切全靠模拟反馈的精妙平衡。这正是 文氏桥振荡器…

Keil下载与串口烧录模式对比图解说明

Keil下载与串口烧录:从开发到量产的程序写入全解析 在嵌入式系统的世界里,代码写得再漂亮,最终也得“刷进去”才算真正落地。而如何把编译好的固件可靠、高效地写入MCU Flash,是每个工程师都绕不开的问题。 面对琳琅满目的工具和…

手把手解析74194四位移位寄存器引脚定义

从零搞懂74194:一块芯片如何让数据“左右横跳”?你有没有想过,那些会流动的LED灯、键盘扫描电路,甚至老式收音机的频道指示条,是怎么实现“一个亮完下一个亮”的?背后藏着一种看似不起眼却极为关键的数字器…

[内网流媒体] 从审计视角看内网服务设计

审计关注什么 谁在什么时候访问了什么资源; 是否有未经授权的访问; 是否符合公司安全/合规要求; 发生问题时能否追溯责任与影响范围。 关键设计点 访问日志 记录时间、IP、路径/流标识、状态码、鉴权结果、User-Agent。 按天滚动,统一时间格式,便于分析与留存。 身份与权…

七段数码管显示数字:基于STM32的硬件连接说明

从点亮一个“8”开始:深入理解STM32驱动七段数码管的底层逻辑 你有没有试过,第一次用单片机点亮一个数字时的那种兴奋? 不是OLED上绚丽的图形,也不是串口打印出的一行数据——而是当你按下复位键,那几个红红的“ 8 …

openmv与stm32通信入门必看:手把手教程(从零实现)

OpenMV与STM32通信实战指南:从零搭建视觉控制系统当你的小车开始“看见”世界想象这样一个场景:你面前的小车不需要遥控,自己就能锁定红色球并追着跑;仓库里的机械臂看到二维码就知道该往哪搬货;机器人通过手势识别理解…

操作指定目录下的文件,对特定参数赋值,接口函数

操作指定目录下的文件,对特定参数赋值,接口函数 操作 /usrdata/root/params.ini文件 并对某些参数赋值 这里为 record_stream参数赋值 #include <stdio.h> #include <string.h> #include <stdlib.h> #include <unistd.h>#define PARAM_FILE "…

MATLAB仿真bp神经网络预测电力负荷 商品形式:程序 实现功能:使用前几日负荷数据预测未来...

MATLAB仿真bp神经网络预测电力负荷 商品形式&#xff1a;程序 实现功能&#xff1a;使用前几日负荷数据预测未来负荷数据 使用bp神经网络 得到误差分析图电力负荷预测这活儿挺有意思的&#xff0c;咱们今天用MATLAB整点实际的。先说说思路&#xff1a;拿前7天的负荷数据当输入…