0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

浅谈void 型指针的高阶用法

multisim 来源:嵌入式客栈 作者: 逸珺 2021-08-09 15:54 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

要比较灵活的使用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指针位置内保存的正确类型的数据进行类型转换。

对于空指针的解引用,你如不信,就来看看栗子:

3147083e-f874-11eb-9bcf-12bb97331649.png

看到了吧,直接解引用编译不过,因为编译器蒙了。

但须注意的是:

不同的编译器对void指针处理是不一样的,如IAR,ANSI C,VC对上述都将出错,而GNU指定“void”的算法操作与“char”一致,因此上述写法在GNU则可以编译

所以做个类型转换,修正如下:

31676c32-f874-11eb-9bcf-12bb97331649.png

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 *指针由这个例子,可以延伸出很多类似的应用

启示:一些语言细节如果深入了解其背后的机理,可以得到很多比较巧妙的应用。

免责声明:本文素材来源网络,版权归原作者所有。如涉及作品版权问题,请与我联系删除。

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • UML
    UML
    +关注

    关注

    0

    文章

    123

    浏览量

    31497
  • C++
    C++
    +关注

    关注

    22

    文章

    2122

    浏览量

    76694
  • void
    +关注

    关注

    0

    文章

    23

    浏览量

    10202

原文标题:void 型指针的高阶用法,你掌握了吗?

文章出处:【微信号:A1411464185,微信公众号:multisim】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    余承东疑再次喊话比亚迪,高阶智驾如何界定?

    电子发烧友网报道(文 / 吴子鹏) 近日,比亚迪在智能化战略发布会上宣布全系搭载 “天神之眼” 高阶智驾系统,这一举措大幅降低了高阶智驾的价格门槛。就连起售价不到 7 万元的入门车型海鸥,也配备了
    的头像 发表于 02-17 01:19 2870次阅读

    C指针的妙用分享

    1、你知道吗?指针其实是个天生的数学家!看这个: #include int main() { int arr[] = {10, 20, 30, 40, 50}; int *p = arr
    发表于 11-17 06:35

    UIAbility组件基本用法说明

    UIAbility组件基本用法 UIAbility组件的基本用法包括:指定UIAbility的启动页面以及获取UIAbility的上下文UIAbilityContext。 指定UIAbility
    发表于 05-16 06:32

    函数指针的六个常见应用场景

    函数指针在嵌入式开发中有着广泛的应用,它让代码更加灵活,减少冗余,提高可扩展性。很多时候,我们需要根据不同的情况动态调用不同的函数,而函数指针正是实现这一需求的重要工具。本文将介绍六个常见的函数指针
    的头像 发表于 04-07 11:58 1117次阅读
    函数<b class='flag-5'>指针</b>的六个常见应用场景

    永磁同步电机自适应高阶滑模Type-2模糊控制

    针对永磁同步电机数学模型不确定问题,提出一种自适应高阶滑模Type-2模糊控制方法。采用积分滑模面二阶滑模控制律,保持传统滑模控制的鲁棒性并实现不含不确定高阶输入输出有限时间稳定;不需要预先确定干扰
    发表于 03-27 11:54

    比亚迪全系车型搭载高阶智驾技术

    2月10日,比亚迪举办智能化战略发布会,重磅发布全民智驾战略。在整车智能战略下,比亚迪构建起天神之眼技术矩阵,其全系车型将搭载高阶智驾技术,其中天神之眼 C首批上市21款车型,覆盖7万级到20万级,包括价格亲民的海鸥,让高阶智驾人人可享,引领汽车行业智能化变革。
    的头像 发表于 02-11 13:45 706次阅读

    void T1_RunTask(void)

    void T1_RunTask(void) { u32 count_th_tl = 0; count_th_tl = ((u16)TH1 << 8) + (u16)TL1
    发表于 02-07 13:36

    指针式万用表测量电压技巧

    在电子维修和电气工程领域,准确测量电压是至关重要的。指针式万用表因其直观的读数和可靠性而广受欢迎。 1. 了解指针式万用表 在开始测量之前,了解万用表的基本构造和功能是必要的。指针式万用表通常有两个
    的头像 发表于 01-23 09:32 2514次阅读

    指针式万用表功能介绍

    在电子工程和电气维修领域,万用表是不可或缺的工具之一。它能够快速测量电路中的电压、电流和电阻等参数,帮助工程师和技术人员诊断和修复电路问题。 指针式万用表的基本原理 指针式万用表的工作原理
    的头像 发表于 01-23 09:12 3923次阅读

    指针式万用表使用指南

    一、指针式万用表简介 指针式万用表是一种传统的电子测量工具,因其表头指针的摆动来显示测量结果而得名。与数字万用表相比,指针式万用表在某些情况下能提供更直观的读数,尤其是在测量快速变化的
    的头像 发表于 01-22 17:25 3261次阅读

    指针式万用表测量精度比较

    指针式万用表的核心是一个可变电阻器(分压器)和一个可动的指针。当测量电压或电流时,通过分压器的电阻值会改变,从而改变通过指针的电流,使指针在刻度盘上移动。
    的头像 发表于 01-22 17:23 2321次阅读

    指针被释放后就变成了空指针

    指针被释放后,是不是就变成了空指针?有好多同学提出了这样的问题。 借用《C专家编程》上面的一段代码,可以很好的解释这个问题。     #include int main(){ char *s
    的头像 发表于 01-22 09:23 645次阅读

    javascript:void(0) 是否影响SEO优化

    使用 javascript:void(0) 确实可能对SEO优化产生负面影响 。以下是关于 javascript:void(0) 对SEO影响的具体分析: 搜索引擎爬虫的理解问题 搜索引擎爬虫(如
    的头像 发表于 12-31 16:08 974次阅读

    javascript:void(0) 的作用是什么

    javascript:void(0) 在 HTML 和 JavaScript 中是一个常见的表达式,主要用来创建一个无操作的链接(通常是 标签)或者阻止默认事件处理。具体来说,它的作用有以下几点
    的头像 发表于 12-31 15:55 4040次阅读

    轻舟智航中高阶智驾解决方案再获项目定点

    近日,轻舟智航宣布其基于地平线征程6M打造的中高阶智驾解决方案「轻舟乘风」正式获得头部新势力车企量产项目定点。目前,轻舟智航基于征程5的智驾方案已取得规模化量产。此次基于征程6M再获车企量产定点,意味着轻舟智航已基于地平线征程平台建立面向中高阶智驾量产的可靠实力。
    的头像 发表于 12-26 15:26 980次阅读