我是靠谱客的博主 积极钢笔,这篇文章主要介绍C语言进阶:指针的进阶(4),现在分享给大家,希望可以做个参考。

函数指针

函数指针的定义

整型指针存放整型的地址;数组指针存放数组的地址;那么类比可得,函数指针存放函数的地址。

显然,函数指针指向函数,存放函数的地址。搞懂函数指针,先了解函数的地址。


&函数名或函数名代表函数地址,与&数组名和数组名略有不同,&函数名和函数名完全一致。

函数的地址必然要放到函数指针里,函数指针的类型该如何写呢?(以Add函数为例)

复制代码
1
2
3
4
5
6
7
8
//整型指针 int* pa = &a; //字符指针 char* pc = &ch; //数组指针 int(*pa)[10] = &arr; //函数指针 - 存放函数地址 int(*pf)(int, int) = &Add;

函数指针的类型

复制代码
1
2
3
4
5
int Add(int x, int y); //1. int(*pf)(int, int) = &Add; //2. int *pf(int, int) = &Add;

倘若,去掉括号int* pf(int, int),pf就变成函数名,返回类型是int*。所以指针必须带括号。

前文已交代,指针,去掉指针名和*就是指针所指向的变量类型。

  • 整型指针,去掉*和指针名,即为整型变量类型int。字符指针,为字符类型char。数组指针,去掉后为数组类型int[10]。
  • 函数指针,去掉*和指针名,即为函数的类型int(int,int)。

总结

  • 去掉指针名pf,即为指针类型int(*)(int, int)
  • 去掉指针名pf和*,即为指针所指向的函数类型为int(int, int)

函数指针的使用

计算机硬件程序经常通过调用地址的方式来调用函数,因此需要使用函数指针调用函数。

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
int Add(int x, int y) { return x + y; } int main() { //1. int(*pf)(int, int) = &Add;//函数指针指向Add函数 //2. int(*pf)(int, int) = Add; //1. int ret = (*pf)(2, 3); //2. int ret = pf(2, 3); printf("%dn", ret); return 0; }

前面已经交代,&函数名和函数名都是函数的地址,完全等价。所以两种初始化函数指针的方式都可以。

既然函数名Add可以直接赋值给函数指针pf,说明二者等价。函数指针名pf不解引用也可以使用,*在此处形同虚设,甚至于不写或写多个都并无大碍,仅为理解。

既然函数名也是函数地址,所以对其解引用也是可以的。我们甚至可以这样写,但仅限娱乐,没有必要。

复制代码
1
2
3
Add(2, 3);//1 (*Add)(2, 3);//2 (*&Add)(2, 3);//3

Example

解释下列代码

复制代码
1
2
3
4
//1. (*(void(*)())0)(); //2. void (*signal(int, void(*)(int)))(int);

1.void(*)()是函数指针类型,放在( )0中,也就是把0强制转换成地址,该地址处存放一个函数其类型为void(*)(void)

2.这样(void(*)())0就变成了指针,指向该地址的函数,且对其解引用访问此函数。

3.(*(void(*)())0)也相当于(*pf),通过函数指针解引用代替函数名,函数名后面带上();,相当于(*pf)();也就是一次不传参的函数调用。

1.signal先和()结合,说明signal为函数名,其后(int, void(*)(int)),为其参数列表。

2.去掉函数名称和参数列表,剩下的void(*)(int)就是返回类型,所以是一次函数声明。

复制代码
1
2
3
4
void (* signal(int, void(*)(int)) ) (int); typedef void(* pf_t)(int);//typedef简化代码 pf_t signal(int, pf_t);

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注靠谱客的更多内容!

最后

以上就是积极钢笔最近收集整理的关于C语言进阶:指针的进阶(4)的全部内容,更多相关C语言进阶:指针内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(123)

评论列表共有 0 条评论

立即
投稿
返回
顶部