c++ 函数指针_C++函数指针深入理解

写在文章前

在写这篇文章之前我想让大家看三个函数的申明:

int p(int a,int b);

int *p(int a,int b);

int (*p)(int a,int b);

我们先看int p(int a,int b)和int *p(int a,int b);我们同时声明这两个函数看系统会报什么错误:无法重载仅按返回类型区分的函数。这说明什么?首先第一:这个两个函数同名,第二,返回值类型不同。因为C++是不支持仅按返回值不同重载函数的,这点大家需要注意。下面这个报错就告诉我们,这两个就是一个返回int的p函数,一个是返回int*的p函数。

5b55c826f5e8fd20134b0714fd19b6f3.png

再看同时声明int p(int a,int b);和int (*p)(int a,int b);会出现什么:声明与1IntelliSense: 声明与 "int p(int a, int b)" (已声明 所在行数: 8) 不兼容。为什么加个括号就不一样了呢?报错是不兼容,说明int *p(int a,int b)和int (*p)(int a,int b);是有很大区别的。

c654e4ea4d298a128b92816a069c6aed.png

int *p(int a,int b)是正常的函数声明,int (*p)(int a,int b)是函数指针的声明

理解函数指针

首先从函数指针的写法上来理解:我们还是看int func(int a, int b)和int (*p)(int a,int b);第一:我们得将int (*p)(int a,int b)看成是在声明一个返回int类型的函数(int a, 和int b)是参数表,(*p)就是与int func(int a, int b)中的func一样,如果大家对python的万物皆对象思想有了解的话应该知道函数也是对象,那么*p就可以看成一个对象。我前面的文章有写过一篇文章讲的是如何理解C++的指针,里面讲到“*”号就是取地址的值,“&”号就是取对象的地址。那么回到函数指针,如果(*p)理解的是对象,那么反过来p指向函数对象的指针变量对吧。在声明时,p是不指向任何函数的,可以理解为指向Null。

67194bf2cc830e526b12d4cb3517fcc1.png

函数指针的使用

首先说一下函数指针有什么用:

假设现在有一个方法funcA();这个方法里面有很多行代码,并且在其中会调用另外某类方法(不确定是哪个,此例是funcB和funcC),如下:

int funcB()

{

return 1 ;

}

int funcC()

{

return 2 ;

}

int funcA(int a,int b)

{

。。。。。。//假设此处有N行代码

int d = funcB(或者funcC) ;

。。。。。。//假设此处有N行代码

return a+b+c + d;

}

如果你现在有个项目可能会在很多处调用funcA(),并且每处需要funcA()中调用的方法都不同,这怎么弄呢,你不会想把funcA()的方法体每次都copy然后改funcB()或者funcC吧?那么,有的同学就会想,能不能讲函数也当成funcA()的参数传进去呢?bingo,恭喜你,有了创造思维,所以函数指针就因此而生。

那么函数指针怎么用呢?

还是拿上面那个例子,那么funcA()函数就的能传递一个函数进来,这就靠函数指针,所以我们先要申明一个函数指针:有几个要点:一、你要传递的函数返回值是什么?而你要传递的函数的参数有哪些。因为你的指针声明是需要跟你要传递的函数一一对应的。这里为了全面体现对funcB()和funcC()做些改变:

int funcB(int a)

{

return a ;

}

int funcC(int a)

{

return a +1;

}

那么声明的函数指针就是int (*p)(int x);

函数funcA()改为:

int funcA(int a,int b,int (*p)(int x))

{

return a+b+p(a);

}

参数里面的int (*p)(int x)就是我们要传的函数的形式参数。

理解函数指针赋值

我们在mian()函数里调用:

p = funcB ;

cout<

p = funcC ;

cout<

这里p = funcB ;和p = funcC 是什么意思呢?首先这里要说一下:每一个函数都占用一段连续的内存单元,它们有一个起始地址。怎么样有没有感觉这种存储方式很熟悉,对,数组也是类似的存储的。那么我们指针跟数组的赋值是怎么样的呢?

int arr[2] = {} ;

int *pArr ;

pArr = arr ;

在数组中数组名就代表着数组的起始地址,那么在指针和函数的赋值中也是同样的道理,函数名就代表着函数内存的起始地址也就是函数入口的地址;那么你pArr[1]这样的写法可以,那么在funcA中我函数指针p(a)这样调用当然也没问题啦!

调用图示如下:

1a4a7d5742c9840e33a3041436497cdc.png
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值