百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

什么是空指针?C语言void指针详解(c语言空指针可以赋值吗)

yund56 2025-05-08 16:41 8 浏览

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\n",ptr1,ptr2);    int i = *ptr2;    printf("%d\n",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吧:

voidmymemset(void *data,int num,int byteSize){    // char就是一个字节,而计算机中是以字节为单位存储的    char *ptr = (char*)data;    int i;    for(i=0;i<byteSize;i++)    {        *ptr=num;        ptr++;    }}intmain(int argc, char *argv[]){    int nums[20];    mymemset(nums,0,sizeof(nums));    int i,len=sizeof(nums)/sizeof(int);    for(i=0;i<len;i++)    {        printf("%d ",nums[i]);    }    printf("\n");    return0;}

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

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

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

typedefstruct _Person{    char *name;    int age;} Person;Person p1;mymemset(&p1,0,sizeof(Person));printf("p1.Age:%d\n",p1.age);

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

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

2

函数指针

2.1 指向函数的指针

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

typedefvoid(*intFunc)(int i);

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

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

voidtest1(int age){    printf("test1:%d\n",age);}intmain(void){        // 声明一个intFunc类型的函数指针    intFunc f1 = test1;    // 执行f1函数指针所指向的代码区    f1(8);    return0;}

最终运行结果如下图所示,执行函数指针f1即执行了其所指向的具体的函数:

2.2 函数指针的基本使用

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

voidforeachNums(int *nums,int len,intFunc func){    int i;    for(i=0;i<len;i++)    {        int num = nums[i];        func(num);    }}voidprintNum(int num){    printf("value=%d\n",num);}

foreachNums函数中,我们定义了一个intFunc函数指针,printNum函数是满足intFunc定义的一个具体的函数。

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

int nums[] = { 1,5,666,23423,223 };    foreachNums(nums,sizeof(nums)/sizeof(int),printNum);

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

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

3

函数指针应用案例

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,int unitSize,int length,compareFunc func){    int i;    char *ptr = (char*)data;    char *max = ptr;        for(i=1;i<length;i++)    {        char *item = ptr+i*unitSize;        //到底取几个字节进行比较是func内部的事情        if(func(item,max)>0)        {            max = item;        }    }    return max;}

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

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

intintDataCompare(void *data1,void *data2){    int *ptr1 = (int*)data1;    int *ptr2 = (int*)data2;    int i1=*ptr1;    int i2=*ptr2;    return i1-i2;}typedefstruct _Dog{    char *name;    int age;} Dog;intdogDataCompare(void *data1,void *data2){    Dog *dog1 = (Dog*)data1;    Dog *dog2 = (Dog*)data2;    return (dog1->age)-(dog2->age);}

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

intmain(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\n",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);    return0;}

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

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

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

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

voidqsort( void * base, size_t num, size_t size, int ( * comparator ) ( constvoid *, constvoid * ) );
    int nums[] = { 3,5,8,7,6 };    qsort(nums,sizeof(nums)/sizeof(int),sizeof(int),intDataCompare);    int i;    for(i=0;i<sizeof(nums)/sizeof(int);i++)    {        printf("%d ",nums[i]);    }    printf("\n");    Dog dogs[] ={{"沙皮",3},{"腊肠",10},{"哈士奇",5},        {"京巴",8},{"大狗",2}};    qsort(dogs,sizeof(dogs)/sizeof(Dog),sizeof(Dog),dogDataCompare);    for(i=0;i<sizeof(dogs)/sizeof(Dog);i++)    {        printf("%s %d ",dogs[i].name,dogs[i].age);    }

那么,快速排序后是否有结果呢?答案是肯定的,我们可以传入各种比较方法,可以升序排序也可以降序排序。

相关推荐

豆包编程能力升级:支持HTML代码实时预览、交互

IT之家3月19日消息,IT之家从豆包官方获悉,豆包宣布AI编程功能迎来三项升级,包括HTML预览、Python运行、生成完整项目。据介绍,目前豆包支持HTML代码实时预览和交互...

1898款游戏!80、90回忆杀,重温旧梦,快速搭建中文DOS游戏服务

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:羊刀仙大家好,我是羊刀仙。本期来介绍一个特别情怀向的游戏项目:chinese-dos-games。这套包含1898款经典中文DOS游戏的合集...

利用 SVG 文件内的 HTML 代码进行网络钓鱼攻击

随着时间的推移,网络钓鱼攻击的技术越来越精妙,旨在欺骗用户并规避安全措施。攻击者会使用欺骗性的URL重定向策略,例如将恶意网站地址附加到看似安全的链接后,在PDF中嵌入链接,以及发送HTML...

aardio + AI 大模型自动编写 Python 代码、网页前端代码的经验与技巧

在AI时代,老式的编程习惯完全被颠覆。原来可能要一大堆插件或工具辛苦堆出来的程序,现在只要把AI调教好了就行。aardio支持调用十几种编程语言,这很适合发挥AI大模型的优势。对于AI...

用AI制作游戏就是如此简单!

很多人不知道如何利用AI提高效率,不知道AI能帮我们做什么,其实可以让我们实现很多自己根本不懂的领域取得直观体验,比如利用DS或者豆包,输入“我想做一个简单的单机俄罗斯方块游戏”,AI会给出phtho...

不会写代码?教你用DeepSeek 三步做出小游戏

如今,借助人工智能技术,哪怕你完全看不懂代码,也能通过DeepSeek制作出属于自己的网页版大鱼吃小鱼游戏。接下来,就为大家详细介绍制作过程。第一步、向DeepSeek描述需求为何要做网页版的...

《暗黑1》被移植成网页游戏 可在浏览器上玩了

《暗黑1》,这款1996年发售的“鼠标杀手”砍杀游戏,现在可以在浏览器上玩了。国外专注暴雪游戏的Rivsoft分享了一个《暗黑1》的共享版本,该版本只包含地下城的头2个地区和三个角色职业中的一个。不...

网页代码过滤 轻松获取专辑目录

通过过滤网页代码,可以将网页上显示不全的长文件名列表完整地提取出来。我有一个含有75个视频文件的《中医诊断学》课件,文件名是以01.RMVB、02.RMVB……75.RMVB这种格式命名的。我希望能找...

IDEA 2021首个大版本发布,Java开发者感动哭了(附新亮点演示)

工欲善其事,必先利其器!就在不久之前,Java领域的开发神器IntelliJIDEA终于迎来2021年的一个重要的大版本更新:IntelliJIDEA2021.1。现如今大量的Java开发者深度...

View Source:在 iOS 上轻松查看网页源代码

在移动互联网时代,移动端的应用和web体验都尤为重要,在PC上有很多web前端工具可以选择,而在移动端貌似就少之又少了,在NEXT出现的ViewSource能帮你在iOS上查看...

当我们《寻找房祖名》,我们能找到什么?

游戏葡萄原创专稿,未经允许请勿转载柯震东,因为在九把刀电影《那些年我们追过的女孩》中饰演男主角柯景腾而走红的台湾影星,在昨天被爆出了和著名演员成龙之子房祖名吸毒被抓的丑闻,一时间相关讨论席卷社交网络。...

多用途游戏娱乐新闻网站HTML5模板

Retnews是一个响应式的HTML新闻,博客,杂志网站模板,可以使用这套前端模板简约很多设计的工作。模板有许多特性适合流行的主题商业、时尚,游戏,娱乐,生活方式、体育、科技、政治、旅行、天气、视频等...

简约好看的个人引导页HTML源码下载

源码介绍一款非常简约好看的个人引导页HTML源码,非常适合个人主页以及个人导航使用,纯HTML不需要数据库,上传服务器即可使用!...

教你如何在微信公共平台上插入小游戏(图文教程)

很多玩微信公共平台的朋友都想在公共平台上面插入几个小游戏,用来跟用户之间互动,这里花生来分享一下如何在微信公共平台上插入游戏,以及如何制作html5微信小游戏。首先是找游戏,总共有三个方法,本人比较倾...

html5重力感应剖析附源码

下面是测试html5重力感应的demohttp://bbs.qietu.com/html/zhongli/http://www.qietu.com/html/f2/qqqianbao/demo2是切图...