要比较灵活的使用c语言实现一些高层级的框架时,需要掌握一些进阶编程技巧,这篇来谈谈void指针的一些妙用。测试环境采用 iar for arm 8.40.1
什么是void指针void指针一般被称为通用指针或叫泛指针。它是c语言关于纯粹地址的一种约定。当某个指针是void型指针时,所指向的对象不属于任何类型。 因为void指针不属于任何类型,则不可以对其进行算术运算,比如自增,编译器不知道其自增需要增加多少。比如char *型指针,自增一定是指针指向的地址加1,short *型指针自增,则偏移2。
在c/c++中,在任意时刻都可以使用其它类型指针来代替void指针,或者用void指针来代替其他类型指针。
由这些特性就可以衍生出很多比较有用的技巧。指针的本质,是其值为一个地址,那么延伸一下:
当使用关键字void声明指针变量时,它将成为通用指针变量。任何数据类型(char,int,float等)的任何变量的地址都可以赋值给void指针变量。
对指针变量的解引用,使用间接运算符*达到目的。但是在使用空指针的情况下,需要转换指针变量以解引用。这是因为空指针没有与之关联的数据类型。编译器无法知道void指针指向的数据类型。因此,要获取由void指针指向的数据,需要使用在void指针位置内保存的正确类型的数据进行类型转换。
对于空指针的解引用,你如不信,就来看看栗子:
看到了吧,直接解引用编译不过,因为编译器蒙了。
但须注意的是:
不同的编译器对void指针处理是不一样的,如iar,ansi c,vc对上述都将出错,而gnu指定“void”的算法操作与“char”一致,因此上述写法在gnu则可以编译
所以做个类型转换,修正如下:
void型指针解引用须做类型指定。
类型转换的时候须注意类型匹配。
另外,如果函数类型可以是任意类型的指针,则需将其参数定义为void *指针,例如string.h中关于内存操作的函数集:
__eff_nenw1nw2 __attributes int memcmp(const void *, const void *,
size_t);
__eff_nenr1nw2r1 __deprec_attrs void * memcpy(void *_restrict,
const void *_restrict,
size_t);
__eff_nenr1nw2r1 __deprec_attrs void * memmove(void *, const void *,
size_t);
__eff_nenr1r1 __deprec_attrs void * memset(void *, int, size_t);
非易失存储管理应用在单片机开发中,往往需要实现数据的非易失存储。所谓非易失存储,就是数据改写后在掉电后仍然能保持。哪些是非易失存储介质呢?比如eeprom,flash等都属于非易失存储介质。
比如一个产品里面有很多各种各样的参数,且分布在各个子系统文件中。举个栗子:
/*模块a中有这样一个结构体需要非易失存储*/typedef struct _t_paras{
int language;/*语言种类*/
char sn[20]; /*产品序列号*/
}t_paras;
t_paras sysparas;
/*模块b中有这样一个结构体需要非易失存储*/typedef struct _t_pid{
float kp;
float ki;
float kd;
float t;
}t_pid;
t_pid pidparas;
面对这样一个需求,要实现非易失存储,我在将底层的eeprom/flash读写函数实现的基础上,将上述应用数据按照一定顺序存储管理。那么更为理想的方式是什么呢?设计一个模块专门负责存储非易失数据。比如:
typedef struct _t_nv_layout{
void * pelement; /*参数地址*/
int length; /*参数长度*/
}t_nv_layout;
/*参数映射表*/
t_nv_layout nvlayout[]={
{&sysparas,sizeof(t_paras)},/*参数映射记录*/
{&pidparas,sizeof(t_pid)},
。。。
};
/*参数映射表记录条数*/#define nv_record_number (sizeof(nvlayout)/sizeof(t_nv_layout))void nv_load(t_nv_layout *playout,int nvaddr,int number);
void nv_store(t_nv_layout *playout,int nvaddr,int number);
将上述设计思想,利用uml描述一下:
在上述基础上,我们只需要设计硬件层抽象,即可设计出一个可行的、比较通用的nv管理子系统,这样设计出的子系统忽略了业务数据,仅仅将其处理为数据,并不关心其业务意义。实现了业务逻辑与后台的隔离解耦。做到了通用性。这里就比较巧妙的利用了void *指针的特性。如果对于该设计思想,在进一步延伸,将底层的抽象在做一层封装,将更细节的底层实现细节隔离抽象,比如:
抽象i2c/spi eeprom,将其对上层的调用接口统一,那么如果你的系统原本是存储在i2c eeprom中,现在做一个新项目,你需要使用另外一种spi接口的eeprom,则只需要实现相应的底层处理函数即可。
将存储介质抽象,比如是eeprom/data flash等。。。
。。。。
那么怎么做到底层抽象呢,我们可以利用函数指针定义统一的接口,具体部署时,只需要将实现函数的指针赋值给对应的函数指针即可,这样就做到了接口的抽象统一。其实这就是驱动模型的一个简易雏形。
总结一下这篇文章引入了一些编程思想,对于单片机/嵌入式进阶编程比较有用:
利用void *指针,将业务数据与底层存储实现了抽象解耦
利用分层抽象实现了代码具有良好的可移植性
利用函数指针实现了c++等高级语言的虚函数定义接口的思想
统一接口底层实现抽象,实现了驱动分层的思想
void *指针由这个例子,可以延伸出很多类似的应用
启示:一些语言细节如果深入了解其背后的机理,可以得到很多比较巧妙的应用。
免责声明:本文素材来源网络,版权归原作者所有。如涉及作品版权问题,请与我联系删除。
彩色滤光片实现OLED彩色化的瑕疵分析
64层3DNAND闪存Intel授权三星、SK海力士、东芝是机遇也是挑战
洲明URMIII系列产品同时获四项国内外顶尖设计大奖
锡焊怎样可以焊铁_锡焊可以焊哪些金属_锡焊焊接的操作要领
STM32和stm32可以超频吗 ?
浅谈void 型指针的高阶用法
政府工作报告中对2021年能源电力有哪些新要求?
多要素气象传感器的型号有哪些?
手机面板市场发展趋势及液晶面板、模组厂家盘点
自动驾驶行业并不是单一的 需要与各行业互相融合发展
人民大学获1.11亿元重大仪器研制专项资助
政策 | 深圳市纯电动物流配送车辆运营资助项目申报指南
飞兆半导体的光耦合器为工业现场总线通信提供出色的隔离性能
中国移动启动FTTx首次集采:GPON国内规模部署
高度智能化生产线,超毅集团抢抓5G产业订单的“新王牌”
明年全国再新建100万个5G基站绝不为过
俄罗斯军工企业技术人员演示多种新型反无人机技术
MCU或MPU上生成AI算法,进行对嵌入式设备操控
口罩合成血液穿透测试仪试验步骤的详细介绍
户外音箱防水透气膜透声膜的特点是什么