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

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

3天内不再提示

面试中的高频问题:指针函数与函数指针,你能完美应对吗?

朱老师物联网大讲堂 2024-06-22 08:11 次阅读

一直觉得C语言较其他语言最伟大的地方就是C语言中的指针,有些人认为指针很简单,而有些人认为指针很难,当然这里的对简单和难并不是等价于对指针的理解程度。为此在这里对C语言中的指针进行全面的总结,从底层的内存分析,彻底让大家明白指针的本质。建议大家静下心来再复习一遍。

一、指针变量

首先要明白指针是一个变量,为此写了如下代码来验证之:

edd7384c-302b-11ef-a655-92fbcf53809c.png

ede28ae4-302b-11ef-a655-92fbcf53809c.png

运行后可以看到a的值被更改了,上面的例子可以清楚的明白指针实质上是一个放置变量地址的特殊变量,其本质仍然是变量。既然指针是变量,那必然会有变量类型,因此这里必须对变量类型做解释。在C语言中,所有的变量都有变量类型,整型、浮现型、字符型、指针类型、结构体、联合体、枚举等,这些都是变量类型。变量类型的出现是内存管理的必然结果,相信读者知道,所有的变量都是保存在计算机的内存中,既然是放到计算机的内存中,那必然会占用一定的空间。问题来了,一个变量会占用多少空间呢,或者说应该分出多少内存空间来放置该变量呢?为了规定这个,类型由此诞生了,对于32位编译器来说,int类型占用4个字节,即32位,long类型占用8字节,即64位。这里简单说了类型主要是为后面引出指针这个特殊性,在计算机中,将要运行的程序都保存在内存中,所有的程序中的变量其实就是对内存的操作。计算机的内存结构较为简单,这里不详细谈论内存的物理结构,只谈论内存模型。将计算机的内存可以想象为一个房子,房子里面居住着人,每一个房间对应着计算机的内存地址,内存中的数据就相当于房子里的人。

edfd3a88-302b-11ef-a655-92fbcf53809c.png

既然指针也是一个变量,那个指针也应该被存放在内存中,对于32位编译器来说,其寻址空间为2^32=4GB,为了能够都操作所有内存(实际上普通用户不可能操作所有内存),指针变量存放也要用32位数即4个字节。
这样就有指针的地址&p,指针和变量的关系可以用如下图表示:ee232a7c-302b-11ef-a655-92fbcf53809c.png

从上图可以看到&p是指针的地址,用来存放指针p,而指针p来存放变量a的地址,也就是&a,还有一个*p在C语言中是解引,意思是告诉编译器取出该地址存放的内容。

ee3e2142-302b-11ef-a655-92fbcf53809c.png

上面提到过关于指针类型的问题,针对32位编译器而言,既然任何指针都只占用4个字节,那为何还需要引入指针类型呢?仅仅是为了约束相同类型的变量么?实际上这里不得不提到指针操作,先思考如下两个操作:

ee5057e0-302b-11ef-a655-92fbcf53809c.png

上面两个操作的意思是不同的,先说下第一种:p+1操作,如下图所示:

ee541d58-302b-11ef-a655-92fbcf53809c.png

对于不同类型指针而言,其 p+1 所指向的地址不同,这个递增取决于指针类型所占的内存大小,而对于 ((unsigned int)p)+1。该意思是将地址 p 所指向的地址的值直接转换为数字,然后 +1 ,这样无论 p 是何种类型的指针,其结果都是指针所指的地址后一个地址。从上述可以看到,指针的存在使得程序员可以相当轻松的操作内存,这也使得当前有些人认为指针相当危险,这一观点表现在 C#Java 语言中,然而实际上用好指针可以极大的提高效率。
下面深入一点来通过指针对内存进行操作,现在我们需要对内存6422216中填入一个数据125,我们可以如下操作:

ee69c72a-302b-11ef-a655-92fbcf53809c.png

当然,上面的代码使用了一个指针,实际上C语言中可以直接利用解引操作对内存进行更方便的赋值,下面说下解引操作*。

二、解引用

所谓解引操作,实际上是对一个地址操作,比如现在想将变量a进行赋值,一般操作是a=125,现在我们用解引操作来完成,操作如下:

ee9adea0-302b-11ef-a655-92fbcf53809c.png

上面可以看到解引操作符为*,这个操作符对于指针有两个不同的意义,当在申明的时候是申明一个指针,而当在使用p指针时是解引操作,解引操作右边是一个地址,这样解引操作的意思就是该地址内存中的数据。这样我们对内存6422216中填入一个数据125就可以使用如下操作:

eeb77ac4-302b-11ef-a655-92fbcf53809c.png

上面需要将6422216数值强制转换为一个地址,这个是告诉编译器该数值是一个地址。值得注意的是上面的所有内存地址不能随便指定,必须是计算机已经分配的内存,否则计算机会认为指针越界而被操作系统杀死即程序提前终止。

三、结构体指针

结构体指针和普通变量指针一样,结构体指针只占4个字节(32位编译器),只不过结构体指针可以很容易的访问结构体类型中的任何成员,这就是指针的成员运算符->。

eedc871a-302b-11ef-a655-92fbcf53809c.png上图中p是一个结构体指针,p指向的是一个结构体的首地址,而p->a可以用来访问结构体中的成员a,当然p->a和*(p)是相同的。

四、强制类型转换

为何要在这里提强制类型转换呢,上面的测试代码可以看到编译器会报很多警告,意思是告诉程序员数据类型不匹配,虽然并不影响程序的正确运行,但是很多警告总会让人感到难受。因此为了告诉编译器代码这里没有问题,程序员可以使用强制类型转换来将一段内存转换为需要的数据类型,例如下面有一个数组a,现在将其强制转换为一个结构体类型stu:

ef1cef80-302b-11ef-a655-92fbcf53809c.png

上面的程序运行结果如下:

ef3af6b0-302b-11ef-a655-92fbcf53809c.png

可以看到a[100]被强制转换为stu结构体类型,当然不使用强制类型转换也是可以的,只是编译器会报警报。

ef4bab7c-302b-11ef-a655-92fbcf53809c.png

上图为程序的示意图,图中数组a[100]的前12个字节被强制转换为了一个struct stu类型,上面仅对数组进行了说明,其它数据类型也是一样的,本质上都是一段内存空间。

五、void指针

为何在这里单独提到空指针类型呢?主要是因为该指针类型很特殊。
void类型很容易让人想到是空的意思,但对于指针而言,其并不是指空,而是指不确定。
在很多时候指针在申明的时候可能并不知道是什么类型或者该指针指向的数据类型有多种再或者程序员仅仅是想通过一个指针来操作一段内存空间。这个时候可以将指针申明为void类型。但是问题来了,由于void类型原因,对于确定的数据类型解引时,编译器会根据类型所占的空间来解引相应的数据,例如int p,那么p就会被编译器解引为p指针的地址的4个字节的空间大小。
但对于空指针类型来说,编译器如何知道其要解引的内存大小呢?先看一段代码:

ef7b18a8-302b-11ef-a655-92fbcf53809c.png

编译上面的程序会发现,编译器报错,无法正常编译。

ef95ff2e-302b-11ef-a655-92fbcf53809c.png

这说明编译器确实是在解引时无法确定*p的大小,因此这里必须告诉编译器p的类型或者*p的大小,如何告诉呢?很简单,用强制类型转换即可,如下:

efe01726-302b-11ef-a655-92fbcf53809c.png

这样上面的程序就可以写为如下:

eff05b04-302b-11ef-a655-92fbcf53809c.png

编译运行后:

f0121730-302b-11ef-a655-92fbcf53809c.png

可以看到结果确实是正确的,也和预期的想法一致。由于void指针没有空间大小属性,因此void指针也没有++操作。

f02c09c4-302b-11ef-a655-92fbcf53809c.png

六、函数指针

6.1 函数指针使用

函数指针在Linux内核中用的非常多,而且在设计操作系统的时候也会用到,因此这里将详细讲解函数指针。既然函数指针也是指针,那函数指针也占用4个字节(32位编译器)。
下面以一个简单的例子说明:

f06b0d36-302b-11ef-a655-92fbcf53809c.png

程序运行结果如下:

f07ad95a-302b-11ef-a655-92fbcf53809c.png

可以看到,函数指针的申明为:

f0a83580-302b-11ef-a655-92fbcf53809c.png

函数指针的解引操作与普通的指针有点不一样。对于普通的指针而言,解引只需要根据类型来取出数据即可,但函数指针是要调用一个函数,其解引不可能是将数据取出,实际上函数指针的解引本质上是执行函数的过程,只是这个执行函数是使用的call指令并不是之前的函数,而是函数指针的值,即函数的地址。
其实执行函数的过程本质上也是利用call指令来调用函数的地址,因此函数指针本质上就是保存函数执行过程的首地址。函数指针的调用如下:

f0c9aee0-302b-11ef-a655-92fbcf53809c.png

为了确认函数指针本质上是传递给call指令一个函数的地址,下面用一个简单例子说明:

f0e940e8-302b-11ef-a655-92fbcf53809c.png

上面是编译后的汇编指令,可以看到,使用函数指针来调用函数时,其汇编指令多了如下:

f10b2f28-302b-11ef-a655-92fbcf53809c.png

分析:第一行mov指令将立即数0x4015c0赋值给寄存器esp+0xc的地址内存中,然后将寄存器esp+0xc地址的值赋值给寄存器eax(累加器),然后调用call指令,此时pc指针将会指向add函数,而0x4015c0正好是函数add的首地址,这样就完成了函数的调用。细心的读者是否发现一个有趣的现象,上述过程中函数指针的值和参数一样是被放在栈帧中,这样看起来就是一个参数传递的过程。因此可以看到,函数指针最终还是以参数传递的形式传递给被调用的函数,而这个传递的值正好是函数的首地址。从上面可以看到函数指针并不是和一般的指针一样可以操作内存,因此作者觉得函数指针可以看作是函数的引用申明。

6.2 函数指针应用

在linux驱动面向对象编程思想中用的最多,利用函数指针来实现封装,下面以一个简单的例子说明:

#include

typedefstructTFT_DISPLAY
{
intpix_width;
intpix_height;
intcolor_width;
void(*init)(void);
void(*fill_screen)(intcolor);
void(*tft_test)(void);

}tft_display;

staticvoidinit(void)
{
printf("thedisplayisinitialed\n");
}

staticvoidfill_screen(intcolor)
{
printf("thedisplayscreenset0x%x\n",color);

}

tft_displaymydisplay=
{
.pix_width=320,
.pix_height=240,
.color_width=24,
.init=init,
.fill_screen=fill_screen,
};

intmain(intargc,char**argv)
{

mydisplay.init();
mydisplay.fill_screen(0xfff);
return0;
}

上面的例子将一个tft_display封装成一个对象,上面的结构体成员中最后一个没有初始化,这在Linux中用的非常多。最常见的是file_operations结构体,该结构体一般来说只需要初始化常见的函数,不需要全部初始化。
上面代码中采用的结构体初始化方式也是在Linux中最常用的一种方式,这种方式的好处在于无需按照结构体的顺序一对一。

6.3 回调函数

有时候会遇到这样一种情况,当上层人员将一个功能交给下层程序员完成时,上层程序员和下层程序员同步工作,这个时候该功能函数并未完成,这个时候上层程序员可以定义一个API来交给下层程序员。而上层程序员只要关心该API就可以了而无需关心具体实现,具体实现交给下层程序员完成即可(这里的上层和下层程序员不指等级关系,而是项目的分工关系)。
这种情况下就会用到回调函数(Callback Function),现在假设程序员A需要一个FFT算法,这个时候程序员A将FFT算法交给程序员B来完成,现在来让实现这个过程:

#include

intInputData[100]={0};
intOutputData[100]={0};

voidFFT_Function(int*inputData,int*outputData,intnum)
{
while(num--)
{

}
}

voidTaskA_CallBack(void(*fft)(int*,int*,int))
{

(*fft)(InputData,OutputData,100);
}

intmain(intargc,char**argv)
{

TaskA_CallBack(FFT_Function);
return0;
}

上面的代码中TaskA_CallBack是回调函数,该函数的形参为一个函数指针,而FFT_Function是一个被调用函数。

可以看到回调函数中申明的函数指针必须和被调用函数的类型完全相同。

本文转载自公众号|芯片之家如有侵权 |联系删除

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

    关注

    180

    文章

    7551

    浏览量

    131880
  • 函数指针
    +关注

    关注

    2

    文章

    54

    浏览量

    3746
  • 指针函数
    +关注

    关注

    0

    文章

    9

    浏览量

    2720
收藏 人收藏

    评论

    相关推荐

    函数指针的使用简述

    函数指针在Linux内核和C语言开发中用的非常多,而且在设计操作系统的时候也会用到,因此这里将详细讲解函数指针。既然函数
    的头像 发表于 09-09 08:56 1778次阅读
    <b class='flag-5'>函数</b><b class='flag-5'>指针</b>的使用简述

    一文详解C语言函数指针与回调函数

    在讲回调函数之前,我们需要了解函数指针
    发表于 10-19 09:34 619次阅读

    函数指针指针函数的概念

    不少朋友会混淆“函数指针”和“指针函数”这两个概念,本文详细介绍一下。
    发表于 03-09 10:49 958次阅读

    C语言中指针函数函数指针的概念及应用示例

    在C语言中,指针函数函数指针是强大且常用的工具。它们允许我们以更灵活的方式处理函数和数据,进而扩展程序的功能。
    发表于 08-16 16:14 393次阅读

    一文详解函数指针与回调函数

    函数指针是指向函数指针变量。它允许我们将函数作为参数传递给其他函数或将
    发表于 08-22 10:36 220次阅读

    函数指针指针函数定义

    函数指针指针函数,C语言学习中最容易混淆的一些概念,好好学习吧
    发表于 01-11 16:44 0次下载

    C语言指针函数函数指针详细介绍

    C语言指针函数函数指针详细介绍。。。。。。。
    发表于 03-04 15:27 5次下载

    c语言函数指针定义,指针函数函数指针的区别

     往往,我们一提到指针函数函数指针的时候,就有很多人弄不懂。下面就由小编详细为大家介绍C语言中函数指针
    发表于 11-16 15:18 3470次阅读

    C语言教程之函数指针变量与指针函数的区别(下篇)

    z=(*pfun)(x,y); //调用函数指针变量,实参x和y的值传递给形参a和b printf(“max=%dn”,z); //形参a和b的值发生交换存储在max函数存储区,因为pfun
    发表于 05-22 07:30 1854次阅读

    ARM的函数指针指针函数的区别和用法

    在学习arm过程中发现这“指针函数”与“函数指针”容易搞错,所以今天,我自己想一次把它搞清楚,找了一些资料,和大家的一些总结,整理到此。和大家分享。
    的头像 发表于 06-20 10:08 3308次阅读

    理解函数指针函数指针数组、函数指针数组的指针

    理解函数指针函数指针数组、函数指针数组的指针
    的头像 发表于 06-29 15:38 1.4w次阅读
    理解<b class='flag-5'>函数</b><b class='flag-5'>指针</b>、<b class='flag-5'>函数</b><b class='flag-5'>指针</b>数组、<b class='flag-5'>函数</b><b class='flag-5'>指针</b>数组的<b class='flag-5'>指针</b>

    指针进阶第五站:函数指针

    定义一个函数指针,指向的函数有两个int形参并且返回一个函数指针,返回的指针指向一个有一个int
    发表于 08-17 15:58 447次阅读

    如何使用函数指针

    在程序中定义一个函数,则编译时系统就会为这段代码分配一段存储空间,这段存储空间的首地址称为函数的地址,且函数名代表的就是这个地址。既然是地址就可以使用一个指针变量来存储这个地址,这个
    的头像 发表于 03-30 15:09 694次阅读

    函数指针指针函数是不是一个东西?

    函数指针的本质是指针,就跟整型指针、字符指针一样,函数指针
    的头像 发表于 01-03 16:35 347次阅读
    <b class='flag-5'>函数</b><b class='flag-5'>指针</b>和<b class='flag-5'>指针</b><b class='flag-5'>函数</b>是不是一个东西?

    函数指针与回调函数的应用实例

    通常我们说的指针变量是指向一个整型、字符型或数组等变量,而函数指针是指向函数函数指针
    的头像 发表于 03-07 11:13 238次阅读
    <b class='flag-5'>函数</b><b class='flag-5'>指针</b>与回调<b class='flag-5'>函数</b>的应用实例