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

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

3天内不再提示

一文揭秘C语言的void指针

工程师进阶笔记 来源:小麦大叔 2023-02-02 17:18 次阅读

1 不能动的“地址”之 void指针

1.1 void指针初探

void *表示一个“不知道类型”的指针,也就不知道从这个指针地址开始多少字节为一个数据。和用int表示指针异曲同工,只是更明确是“指针”。

因此void *只能表示一个地址,不能用来&取值,也不能++和--移动指针,因此不知道多少字节是一个数据单位。

intnums[]={3,5,6,7,9};
void*ptr1=nums;
//inti=*ptr1;//对于void指针没法直接取值
int*ptr2=(int*)nums;
printf("%d,%d
",ptr1,ptr2);
inti=*ptr2;
printf("%d
",i);

从输出结果可以看出,无论是无类型的void指针还是int类型指针,指向的地址都是一样的:

89ca1ff4-a2d8-11ed-bfe3-dac502259ad0.png

PS:void *就是一个不能动的“地址”,在进行&、移动指针之前必须转型为类型指针。

1.2 void指针的用途

89d7eaf8-a2d8-11ed-bfe3-dac502259ad0.png

这里我们看一下我们之前了解的memset函数,其第一个参数就是一个void指针,它可以帮我们屏蔽各种不同类型指针的差异。 如下面代码所示,我们既可以传入一个int类型数组的指针,也可以传入一个char类型数组的指针:

intnums[20];
memset(nums,0,sizeof(nums));
charchs[2];
memset(chs,0,sizeof(chs));

那么,我们也可以试着自己动手模拟一下这个memset函数,暂且命名为mymemset吧:

voidmymemset(void*data,intnum,intbyteSize)
{
//char就是一个字节,而计算机中是以字节为单位存储的
char*ptr=(char*)data;
inti;
for(i=0;i

在这个mymemset函数中,我们利用void指针接收不同类型的指针,利用char类型(一个字节)逐个字节读取内存中的每一个字节,最后依次填充指定的数字。

由于char类型是一个具体类型,所以可以使用++或者--进行指针的移动。

对于结构体类型,也可以使用我们的mymemset函数:

typedefstruct_Person
{
char*name;
intage;
}Person;

Personp1;
mymemset(&p1,0,sizeof(Person));
printf("p1.Age:%d
",p1.age);

最终的运行结果如下图所示:

89ec4638-a2d8-11ed-bfe3-dac502259ad0.png

void *的用途:在只知道内存,但是不知道是什么类型的时候。

2 函数指针

2.1 指向函数的指针

我们可以在C中轻松地定义一个函数指针:

typedefvoid(*intFunc)(inti);

这里我们定义了一个无返回值的,只有一个int类型参数的函数指针intFunc。

我们可以在main函数中使用这个函数指针来指向一个具体的函数(这个具体的函数定义需要和函数指针的定义一致):

voidtest1(intage){

printf("test1:%d
",age);
}

intmain(void){

//声明一个intFunc类型的函数指针
intFuncf1=test1;
//执行f1函数指针所指向的代码区
f1(8);
return0;
}
最终运行结果如下图所示,执行函数指针f1即执行了其所指向的具体的函数:

89fece66-a2d8-11ed-bfe3-dac502259ad0.png

2.2 函数指针的基本使用

这里我们通过一个小案例来对函数指针做一个基本的使用介绍。相信大部分的C#Java程序员都很熟悉foreach,那么我们就来模拟foreach对int数组中的值进行不同的处理。具体体现为for循环的代码是复用的,但是怎么处理这些数据不确定,因此把处理数据的逻辑由函数指针指定。

voidforeachNums(int*nums,intlen,intFuncfunc)
{
inti;
for(i=0;i
在foreachNums函数中,我们定义了一个intFunc函数指针,printNum函数是满足intFunc定义的一个具体的函数。

下面我们在main函数中将printNum函数作为函数指针传递给foreachNums函数。

intnums[]={1,5,666,23423,223};
foreachNums(nums,sizeof(nums)/sizeof(int),printNum);
最终运行的结果如下图所示:

8a108c14-a2d8-11ed-bfe3-dac502259ad0.png

通过函数指针,我们可以屏蔽各种具体处理方法的不同,也就是将不确定的因素都依赖于抽象,这也是面向抽象或面向接口编程的精髓。

三、函数指针应用案例

3.1 计算任意类型的最大值

(1)定义函数指针及getMax主体:

typedefint(*compareFunc)(void*data1,void*data2);
// getMax 函数参数说明:
//data待比较数据数组的首地址,uniteSize单元字节个数
// length:数据的长度。{1,3,5,6}:length=4
//比较data1和data2指向的数据做比较,
//如果data1>data2,则返回正数
void*getMax(void*data,intunitSize,intlength,compareFuncfunc)
{
inti;
char*ptr=(char*)data;
char*max=ptr;

for(i=1;i0)
{
max=item;
}
}

returnmax;
}
这里可以看到,在getMax中到底取几个字节去比较都是由compareFunc所指向的函数去做,getMax根本不用关心。

(2)定义符合函数指针定义的不同类型的函数:

intintDataCompare(void*data1,void*data2)
{
int*ptr1=(int*)data1;
int*ptr2=(int*)data2;

inti1=*ptr1;
inti2=*ptr2;

returni1-i2;
}

typedefstruct_Dog
{
char*name;
intage;
}Dog;

intdogDataCompare(void*data1,void*data2)
{
Dog*dog1=(Dog*)data1;
Dog*dog2=(Dog*)data2;

return(dog1->age)-(dog2->age);
}

(3)在main函数中针对int类型和结构体类型进行调用:

intmain(intargc,char*argv[])
{
//test1:int类型求最大值
intnums[]={3,5,8,7,6};
int*pMax=(int*)getMax(nums,sizeof(int),sizeof(nums)/sizeof(int),
intDataCompare);
intmax=*pMax;
printf("%d
",max);
//test2:结构体类型求最大值
Dogdogs[]={{"沙皮",3},{"腊肠",10},{"哈士奇",5},
{"京巴",8},{"大狗",2}};
Dog*pDog=(Dog*)getMax(dogs,sizeof(Dog),
sizeof(dogs)/sizeof(Dog),dogDataCompare);
printf("%s=%d",pDog->name,pDog->age);

return0;
}

最终运行结果如下图所示:

8a221a42-a2d8-11ed-bfe3-dac502259ad0.png

3.2 C 中自带的qsort函数—自定义排序

qsort包含在头文件中,此函数根据你给的比较条件进行快速排序,通过指针移动实现排序。排序之后的结果仍然放在原数组中。

使用qsort函数必须自己写一个比较函数。我们可以看看qsort函数的原型:

voidqsort(void*base,size_tnum,size_tsize,int(*comparator)(constvoid*,constvoid*));
intnums[]={3,5,8,7,6};
qsort(nums,sizeof(nums)/sizeof(int),sizeof(int),intDataCompare);
inti;
for(i=0;i
那么,快速排序后是否有结果呢?答案是肯定的,我们可以传入各种比较方法,可以升序排序也可以降序排序。

8a3678d4-a2d8-11ed-bfe3-dac502259ad0.png

审核编辑:汤梓红

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

    关注

    180

    文章

    7533

    浏览量

    128813
  • 函数
    +关注

    关注

    3

    文章

    3903

    浏览量

    61310
  • 指针
    +关注

    关注

    1

    文章

    473

    浏览量

    70364
  • void
    +关注

    关注

    0

    文章

    23

    浏览量

    9783
  • 函数指针
    +关注

    关注

    2

    文章

    53

    浏览量

    3737

原文标题:揭秘 C 语言的 void 指针

文章出处:【微信号:工程师进阶笔记,微信公众号:工程师进阶笔记】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    void指针简述

    为何在这里单独提到空指针类型呢?主要是因为该指针类型很特殊。void类型很容易让人想到是空的意思,但对于指针而言,其并不是指空,而是指不确定。在很多时候
    的头像 发表于 09-09 08:57 1486次阅读
    <b class='flag-5'>void</b><b class='flag-5'>指针</b>简述

    C语言void指针的基本认识及典型应用

    C语言中,*类型就是指针类型,比如 int *p,double *q,虽然是不一样的指针,但是大小却一样sizeof(p) == sizeof(q),因为它们都是同一种类型*类型的。C语言
    发表于 02-01 09:26 849次阅读

    知道PHP`引用`和C语言`指针`的区别?

    PHP`引用`和C语言`指针`的区别是什么
    发表于 10-27 08:27

    c语言指针详解

    。9、空指针 指向空,或者说不指向任何东西。在C语言中,我们让指针变量赋值为NULL表示个空指针
    发表于 03-26 09:51

    谈谈void指针些妙用

    关注+星标公众号,不错过精彩内容转自 | 嵌入式客栈要比较灵活的使用C语言实现些高层级的框架时,需要掌握些进阶编程技巧,这篇来谈谈void
    发表于 01-26 07:20

    什么是void指针void指针有何功能

    [导读] 要比较灵活的使用C语言实现些高层级的框架时,需要掌握些进阶编程技巧,这篇来谈谈void指针
    发表于 02-21 06:01

    彻底搞懂C语言指针

    以后,在C程序同样会在内存中划分出块空间,用于存放指针类型的值这个指针类型的值就是个内存地址。这里需要特别区分三个概念,
    发表于 07-22 14:48

    C语言voidvoid指针深层探索

    1.概述 本文将对void关键字的深刻含义进行解说,并详述voidvoid指针类型的使用方法与技巧。 2.void的含义
    发表于 09-21 11:36 166次下载

    深入了解void指针背后的机理

    当使用关键字void声明指针变量时,它将成为通用指针变量。任何数据类型(char,int,float等)的任何变量的地址都可以赋值给void指针
    发表于 05-06 09:49 1226次阅读

    浅谈void指针的高阶用法

    要比较灵活的使用C语言实现一些高层级的框架时,需要掌握一些进阶编程技巧,这篇来谈谈void指针的一些妙用。测试环境采用 IAR for ARM 8.40.1 什么是void
    的头像 发表于 08-09 15:54 1383次阅读
    浅谈<b class='flag-5'>void</b> 型<b class='flag-5'>指针</b>的高阶用法

    STM32编程:void指针高阶用法举例,设计一个通用掉电存储子系统

    [导读] 要比较灵活的使用C语言实现一些高层级的框架时,需要掌握一些进阶编程技巧,这篇来谈谈void指针的一些妙用。测试环境采用 IAR for ARM 8.40.1什么是void
    发表于 12-27 18:47 10次下载
    STM32编程:<b class='flag-5'>void</b><b class='flag-5'>指针</b>高阶用法举例,设计一个通用掉电存储子系统

    STM32编程:void指针高阶用法

    [导读] 要比较灵活的使用C语言实现一些高层级的框架时,需要掌握一些进阶编程技巧,这篇来谈谈void指针的一些妙用。测试环境采用 IAR fo...
    发表于 02-07 11:14 2次下载
    STM32编程:<b class='flag-5'>void</b><b class='flag-5'>指针</b>高阶用法

    C语言-void类型作为万能指针类型

    空类型`。 void abc; //这是错误的 但是`void`是可以定义指针的,`void*`表示万能型指针类型,可以与任何
    的头像 发表于 08-14 10:04 1479次阅读

    什么是void指针

    C语言中,*类型就是指针类型,比如 int *p,double *q,虽然是不一样的指针,但是大小却一样sizeof(p) == sizeof(q),因为它们都是同一种类型*类型的。C语言
    的头像 发表于 02-13 15:02 1046次阅读

    void指针

    C语言中,*类型就是指针类型,比如 int *p,double *q,虽然是不一样的指针,但是大小却一样sizeof(p) == sizeof(q),因为它们都是同一种类型*类型的。C语言
    的头像 发表于 02-20 15:09 547次阅读