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相当于一个跳板的作用,我们经常把这样的数组叫做转移表!


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


相关文章
|
19天前
|
存储 C语言
【C语言篇】深入理解指针3(附转移表源码)
【C语言篇】深入理解指针3(附转移表源码)
30 1
|
19天前
|
存储 程序员 编译器
【C语言】指针篇-简单快速了解指针-必读指南(1/5)
【C语言】指针篇-简单快速了解指针-必读指南(1/5)
|
4天前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
4天前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
4天前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
5天前
|
C语言
C语言指针(3)
C语言指针(3)
9 1
|
5天前
|
C语言
C语言指针(2)
C语言指针(2)
9 1
|
11天前
|
存储 搜索推荐 C语言
深入C语言指针,使代码更加灵活(二)
深入C语言指针,使代码更加灵活(二)
|
11天前
|
存储 程序员 编译器
深入C语言指针,使代码更加灵活(一)
深入C语言指针,使代码更加灵活(一)
|
11天前
|
C语言
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)