一文揭秘C语言的void指针

1 不能动的“地址”之 void指针
1.1 void指针初探
void *表示一个“不知道类型”的指针,也就不知道从这个指针地址开始多少字节为一个数据。和用int表示指针异曲同工,只是更明确是“指针”。
因此void *只能表示一个地址,不能用来&取值,也不能++和--移动指针,因此不知道多少字节是一个数据单位。
    int nums[] = {3,5,6,7,9};    void* ptr1 = nums;    //int i = *ptr1; // 对于void指针没法直接取值    int* ptr2 = (int*)nums;    printf(%d,%d,ptr1,ptr2);    int i = *ptr2;    printf(%d,i);  
从输出结果可以看出,无论是无类型的void指针还是int类型指针,指向的地址都是一样的:
ps:void *就是一个不能动的“地址”,在进行&、移动指针之前必须转型为类型指针。
1.2 void指针的用途
这里我们看一下我们之前了解的memset函数,其第一个参数就是一个void指针,它可以帮我们屏蔽各种不同类型指针的差异。 如下面代码所示,我们既可以传入一个int类型数组的指针,也可以传入一个char类型数组的指针:
    int nums[20];    memset(nums,0,sizeof(nums));    char chs[2];    memset(chs,0,sizeof(chs));  
那么,我们也可以试着自己动手模拟一下这个memset函数,暂且命名为mymemset吧:
void mymemset(void *data,int num,int bytesize){    // char就是一个字节,而计算机中是以字节为单位存储的    char *ptr = (char*)data;    int i;    for(i=0;i 在这个mymemset函数中,我们利用void指针接收不同类型的指针,利用char类型(一个字节)逐个字节读取内存中的每一个字节,最后依次填充指定的数字。
由于char类型是一个具体类型,所以可以使用++或者--进行指针的移动。
对于结构体类型,也可以使用我们的mymemset函数:
typedef struct _person{    char *name;    int age;} person;person p1;mymemset(&p1,0,sizeof(person));printf(p1.age:%d,p1.age);  
最终的运行结果如下图所示:
void *的用途:在只知道内存,但是不知道是什么类型的时候。
2 函数指针
2.1 指向函数的指针
我们可以在c中轻松地定义一个函数指针:
typedef void (*intfunc)(int i);  
这里我们定义了一个无返回值的,只有一个int类型参数的函数指针intfunc。
我们可以在main函数中使用这个函数指针来指向一个具体的函数(这个具体的函数定义需要和函数指针的定义一致):
void test1(int age){    printf(test1:%d,age);}int main(void){        // 声明一个intfunc类型的函数指针    intfunc f1 = test1;    // 执行f1函数指针所指向的代码区    f1(8);    return 0;}最终运行结果如下图所示,执行函数指针f1即执行了其所指向的具体的函数:
2.2 函数指针的基本使用
这里我们通过一个小案例来对函数指针做一个基本的使用介绍。相信大部分的c#或java程序员都很熟悉foreach,那么我们就来模拟foreach对int数组中的值进行不同的处理。具体体现为for循环的代码是复用的,但是怎么处理这些数据不确定,因此把处理数据的逻辑由函数指针指定。
void foreachnums(int *nums,int len,intfunc func){    int i;    for(i=0;idata2,则返回正数void *getmax(void *data,int unitsize,int length,comparefunc func){    int i;    char *ptr = (char*)data;    char *max = ptr;        for(i=1;i0)        {            max = item;        }    }    return max;}这里可以看到,在getmax中到底取几个字节去比较都是由comparefunc所指向的函数去做,getmax根本不用关心。  
(2)定义符合函数指针定义的不同类型的函数:
int intdatacompare(void *data1,void *data2){    int *ptr1 = (int*)data1;    int *ptr2 = (int*)data2;    int i1=*ptr1;    int i2=*ptr2;    return i1-i2;}typedef struct _dog{    char *name;    int age;} dog;int dogdatacompare(void *data1,void *data2){    dog *dog1 = (dog*)data1;    dog *dog2 = (dog*)data2;    return (dog1->age)-(dog2->age);}  
(3)在main函数中针对int类型和结构体类型进行调用:
int main(int argc, char *argv[]){    // test1:int类型求最大值    int nums[] = { 3,5,8,7,6 };    int *pmax = (int *)getmax(nums,sizeof(int),sizeof(nums)/sizeof(int),        intdatacompare);    int max = *pmax;    printf(%d,max);    // test2:结构体类型求最大值    dog dogs[] ={{沙皮,3},{腊肠,10},{哈士奇,5},        {京巴,8},{大狗,2}};    dog *pdog = (dog *)getmax(dogs,sizeof(dog),        sizeof(dogs)/sizeof(dog),dogdatacompare);    printf(%s=%d,pdog->name,pdog->age);    return 0;}  
最终运行结果如下图所示:
3.2 c 中自带的qsort函数—自定义排序
qsort包含在头文件中,此函数根据你给的比较条件进行快速排序,通过指针移动实现排序。排序之后的结果仍然放在原数组中。
使用qsort函数必须自己写一个比较函数。我们可以看看qsort函数的原型:
void qsort ( void * base, size_t num, size_t size, int ( * comparator ) ( const void *, const void * ) );    int nums[] = { 3,5,8,7,6 };    qsort(nums,sizeof(nums)/sizeof(int),sizeof(int),intdatacompare);    int i;    for(i=0;i

汽车线束端子压接电压降的产生原因
基于LM3S101处理器的温度测量模块设计
如何使用嵌入式 STM32F769I-DISCO 麦克风
小米Note10核心配置信息曝光搭载骁龙855 Plus支持40W快充
多参数水质分析仪概述、用途、原理及参数
一文揭秘C语言的void指针
光线强弱检测实验
如何在高共模电压下测量小差分电压
高性能单电源 ARM7 微控制器介绍
现代示波器和数字化仪的使用规则
智能魔镜,让你上瘾的镜子显示屏
“缺芯”环境下,国产MCU迎来窗口期
电梯安全无线监控方案
LDO基础知识:电源抑制比
通过智能家居魔镜的应用来让家里的设备互动起来
大兴机场积极尝试引入探鸟雷达技术
如何将弱电、安防、监控和综合布线故障和失误减至最低
捷荣技术:预计2023年半年度亏损4055万元—5272万元
联想发布可折叠笔记本 折叠角度也可以调节
经纬恒润正式升级为AUTOSAR高级合作伙伴