C语言进阶第三篇【指针进阶】(上)

简介: C语言进阶第三篇【指针进阶】(上)

前言:Hello!我是@每天都要敲代码。上一期我们一起学习了:字符指针的使用、指针数组和数组指针的理解、数组和指针的传参以及补充的一些练习题目<<传送门>>;还没有掌握的一定要去学习在看下面的知识点,因为指针的知识点都是连贯的!这一期我们将继续学指针剩余的内容,一起加油吧!


1. 函数指针

1.1 函数指针的理解和写法

我们先根据前面所学梳理清楚函数指针是什么?


int* p          整型指针---指向整型的指针---存放的是整型变量的地址

char* pc     字符型指针---指向字符型的指针---存放的是字符变量的地址

int(*p)[10]  数组指针---指向数组的指针---存放的是数组的地址

----------       函数指针---肯定就是指向函数的指针---存放的是函数的地址


我们推导出了函数指针的意义,那么怎么书写呢?


26cff1e4d3784d4ab2923e5e2afaf635.png


这里我只说两点:


1、我们发现数组指针和函数指针的书写形式很类似:int (*parr)[10] = &arr


                                                                                    int (*pf)(int,int) = &ADD


2、我们打印&ADD和ADD的地址居然也是一样的,也类似于数组&arr和arr;我们来做一下比较:


(1)数组名 != &数组名;两者不等价,前者取出来的是首元素的地址;后者取出来的是整个数组的地址

(2)函数名 == &函数名;两者是等价的,因为对于函数没有什么首元素的地址这一说


注意:()的优先级也要高于*号的,所以必须加上()来保证pf先和*结合


练习:有了上面的理解,我们不妨就拿一个例题来来练练手,看你是否真正理解掌握了函数指针的写法,例如:void test(char* str),函数指针怎么书写呢?void (*pt)(char*) = &test


1.2 函数指针的调用

我们还是先从已学的知识入手,然后剖析函数指针的调用


baeeea3753d345da968ed7910ba635a0.png


这里解释一下两点:


1、int (*pf)(int, int) = &ADD   等价于     int (*pf)(int, int) = ADD


2、int sum = (*pf)(2, 3)           等价于    int sum = pf (2, 3);这里的解应用*没有实际意思,是摆设


1.3 两段有趣的代码

1.3.1 代码1 (*(void (*)())0)()

( *( void (*)() )0 )() 我们先根据自己的想法通俗理解一下:将0强制类型转换为函数指针,然后解应用,在调用。


解析:


1、从数字0开始着手,想让数字0作为地址,肯定要是一种指针类型才可以;

2、把0强制类型转换为函数指针类型,0作为函数的地址;

3、0放到函数指针类型里是无参的,返回类型是void;

4、然后开始调用,我们解应用括号括起来,()代表我们调用它,参数什么都不传;

总结:


调用0地址处的函数;该函数无参,返回类型是void

1、void(*)()---函数指针类型

2、(void (*)())0---对0进行强制类型转换,被解释为一个函数地址

3、*(void (*)())0---对0地址进行解引用操作

4、(*(void (*)())0)()---调用0地址处的函数


1.3.2 代码2 void (*signal(int,void(*)(int)))(int)

void (* signal(int,void(*)(int)) )(int)我们还是要一步步拆分去理解


1、signal和()先结合,说明signal是函数名;

2、signal函数的第一个参数的类型是int第,二个参数的类型是函数指针;该函数指针,指向一个参数为int,返回类型为void的函数;

3、signal函数的返回类型也是一个函数指针;该函数指针,指向一个参数为int,返回类型为void的函数;所以,signal是一个函数的声明;

4、void (*)(int) signal(int,void(*)(int))我们也可以这样理解,signal函数的返回类型是一个函数指针;但是语法上不能这样写,只能把返回类型写中间;


简化:signal是一个参数函数指针,它的返回类型又是一个函数指针;那我们怎么优化呢?


利用typedef-对类型重定义:typedef void(*) (int) pfun_t;对void(*)(int)的函数指针类型进行重定义为pfun_t;但是这种语法是不支持的,我们要把pfun_t写到中间:


typedef void(*pfun_t) (int);重名之后上面代码就可以拆分成:


void (*)(int) signal(int,void(*)(int)) 和typedef void(*pfun_t) (int)等价于pun_t signal(int,punt)


是不是更加的容易理解了?


2. 函数指针数组

2.1 函数指针的理解和写法

我们先给出对于函数指针数组的理解:存放函数指针的数组;


我们在一起回顾一下整型指针数组:整型指针 int*,整型指针数组 int* arr[10];那么函数指针数组怎么定义和使用呢?我们通过一段代码的形式去理解。

4a5cf3cccdfd495ab6139d5dfd9316bf.png



2.2 函数指针的实际应用

那么函数指针在实际写代码中有什么应用呢?我们就通过一个典型的计算器来对比学习;比如:我们要实现加、减、乘、除。


2.2.1 普通方法实现计算器

b5c5293acb464309abd32ece9104bc65.png


我们从switch中就可以看出,代码很冗余。那我们思考一下:


1、如果把 printf("请输入两个操作数:>");scanf_s("%d %d", &x, &y);这两句代码全部提炼出去放到switch最前。printf("ret=%d\n", ret);这句代码提炼出区放到后面不久可以避免代码冗余了?


2、但是这样会出现我们问题,当我们选择5错误或者选择0退出时,它还是要让我们输入两个操作数才能退出,这就很怪异;包括打印也是,我们只有正确调用了函数,才能打印;所以这种方法并不可取;怎么办呢?下面我们就用函数指针数组来优化这个代码!


2.2.2 函数指针数组实现计算器


af239986fea44ab79cc23680e5e2cc8d.png

我们利用函数指针数组来实现计算器,即解决了代码冗余的问题,而且还让代码更加的简练了,是不是很妙?这里函数指针数组parr相当于一个跳板的作用,我们经常把这样的数组叫做转移表!


函数指针数组的用途:转移表


相关文章
|
4天前
|
C语言
指针进阶(C语言终)
指针进阶(C语言终)
|
4天前
|
C语言
指针进阶(回调函数)(C语言)
指针进阶(回调函数)(C语言)
|
4天前
|
存储 C语言 C++
指针进阶(函数指针)(C语言)
指针进阶(函数指针)(C语言)
|
4天前
|
编译器 C语言
指针进阶(数组指针 )(C语言)
指针进阶(数组指针 )(C语言)
|
2天前
|
Java 程序员 Linux
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
9 0
|
3天前
|
C语言
C语言中的函数指针、指针函数与函数回调
C语言中的函数指针、指针函数与函数回调
6 0
|
3天前
|
存储 C语言
C语言中的多级指针、指针数组与数组指针
C语言中的多级指针、指针数组与数组指针
6 0
|
3天前
|
存储 C语言
C语言数组指针详解与应用
C语言数组指针详解与应用
10 0
|
3天前
|
存储 C语言
C语言中的指针
C语言中的指针
8 0
|
2月前
|
C语言
C语言---指针进阶
C语言---指针进阶
29 0