指针进阶(详解)
创始人
2024-05-08 23:11:30
0

指针进阶

  • 一.字符指针
  • 二.指针数组
    • 1.一次打印多个字符串
    • 2.模拟二维数组
  • 三.数组指针
    • 1.定义
    • 2.应用
  • 四.函数指针
  • 五.函数指针数组
  • 六.指向函数指针数组的指针
  • 七.回调函数

在这里插入图片描述

在开始这篇之前,前面有两篇指针初阶,如果需要的话可以去看看哟!指针初阶1,指针初阶2

一.字符指针

在这里插入图片描述

对于一个字符串,那么指针又该如何指向呢?

在这里插入图片描述

毫无疑问p里存的是地址,那它存了abcdefg的所有地址吗?当然不是,因为它根本存不下(一个指针在32为机器下只有4个字节,如果不太明白可以看看指针初阶)。实际上,它存的是首字符a的地址。字符串在内存里是连续存放的,所以只需要找到首元素地址就可以啦。ps:这和数组很像,事实上处理字符串时也可以按照下标来处理。

在这里插入图片描述

追加一个小知识:字符串可以直接使用%s打印是因为它有\0作为结束标志。但如果全部是整数的话,没有结束标志就只能通过下标挨个打印。

一道面试题

在这里插入图片描述

在这里插入图片描述

首先,str1和str2是两个数组,需要开辟两个空间,而数组名代表首元素地址,空间不同首元素地址自然不同,故str1!=str2。但是str3和str4是指针变量,里面存的都是h的地址并且hello,bite是一个常量字符串永远不能被改变。所以此时编译器就会认为既然都一样且没法被改变,那么就只会开辟一个空间,str3和str4都指向同一空间,故str3==str4。(可以理解为编译器的一种规定)

二.指针数组

存放整形的数组->整形数组,存放字符的数组->字符数组,存放指针的数组->指针数组。

1.一次打印多个字符串

在这里插入图片描述

在这里插入图片描述

我们如何打印呢?很简单,因为数组内每个元素存的都是各个字符串首元素地址,所以我们只需要依靠首元素地址就能打印出所有字符串。

在这里插入图片描述
在这里插入图片描述

2.模拟二维数组

在这里插入图片描述

它的打印需要使用两个for嵌套,因为它不能像字符串那样打印一串,这里其实也就是用一维数组模拟二维数组(但不同的是二维数组每一行每一列都是连续存放的,而这里每一行不一定连续)

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

三.数组指针

1.定义

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

追加个小知识:【】的优先级高于*所以别忘了加括号哦。

ps:如果pa先与*结合就说明是个指针,如果先于【】结合就说明是个数组。

pps:这里的&arr是整个数组的地址,也就是如果+1的话是跳过整个数组。注意与数组名区分开。

2.应用

1.强行使用

在这里插入图片描述

这样写实际上很别扭,当然实际上一般也没人会怎么写。所以它的真正作用体现在二维数组里。

2.正常使用

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

一些练习

在这里插入图片描述

四.函数指针

指向函数的指针。

一段代码

首先,函数名=&函数,故函数名就是函数地址(如果需要验证的话,写一个函数再用%p输出就可以了,这里就不再累述了)

在这里插入图片描述

分析这个指针的构成,首先是一个指针故pf应该先与*结合,其次指向一个函数,后面就应该接上(参数),最后该函数是int类型,故再前面写上int。将这3个部分串起来就是如上指针。

接下来使用pf指针调用Add函数。

在这里插入图片描述

在这里插入图片描述

首先对pf进行解引用,解引用后就是原函数。也就是*pf=Add,接下来是传参(参数1,参数2),这样就完成了调用功能。实际上,在调用时前面的星号是没有作用的,因为如我们平常调用函数Add(2,3),pf里存的本身就是Add,所以不需要解引用。

在这里插入图片描述

代码一

在这里插入图片描述

这道题的关键就是把0当成地址来看。

在这里插入图片描述

该代码是一次函数调用,调用0地址处的函数

ps:以上代码出自c语言缺陷与陷阱

代码二

在这里插入图片描述

依然从名字signal入手。

在这里插入图片描述

1.该代码是一次函数的声明。
2.声明的函数名字是signal
3.参数有两个,第一个是int类型,第二个是函数指针类型
4.signal函数的返回值是一个函数指针

五.函数指针数组

数组是一个存放相同类型数据的存储空间,那我们已经学习了指针数组,比如:int arr[10];数组的每个元素是int。那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组,那函数指针的数组如何定义呢?

在这里插入图片描述

使用:简单的计算器

int add(int a, int b){return a + b;
}
int sub(int a, int b){return a - b;
}
int mul(int a, int b){return a * b;
}
int div(int a, int b){return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;do{printf("*************************\n");printf("*** 1:加法      2:减法***\n");printf("*** 3:乘法      4:除法*** \n");printf("**********0.退出*********\n");printf("*************************\n");printf("请选择:");scanf("%d", &input);switch (input){case 1:printf("输入操作数:");scanf("%d %d", &x, &y);ret = add(x, y);printf("ret = %d\n", ret);break;case 2:printf("输入操作数:");scanf("%d %d", &x, &y);ret = sub(x, y);printf("ret = %d\n", ret);break;case 3:printf("输入操作数:");scanf("%d %d", &x, &y);ret = mul(x, y);printf("ret = %d\n", ret);break;case 4:printf("输入操作数:");scanf("%d %d", &x, &y);ret = div(x, y);printf("ret = %d\n", ret);break;case 0:printf("退出程序\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

这个代码很简单就不多说了。实际上我们可以看到这个代码很冗长繁琐,如果在之后我们需要给计算器添加新的功能的话,毫无疑问case会加长,我们能不能有一种方法避免这种写法呢?

int add(int a, int b){return a + b;
}
int sub(int a, int b){return a - b;
}
int mul(int a, int b){return a * b;
}
int div(int a, int b){return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //使用数组去除冗长的调用while (input){printf("*************************\n");printf("*** 1:加法      2:减法***\n");printf("*** 3:乘法      4:除法*** \n");printf("**********0.退出*********\n");printf("*************************\n");printf("请选择:");scanf("%d", &input);if (0 == input){printf("退出程序。\n");break;}if ((input <= 4 && input >= 1)){printf("输入操作数:");scanf("%d %d", &x, &y);ret = (*p[input])(x, y);}elseprintf("输入有误\n");printf("ret = %d\n", ret);}return 0;
} 

使用函数指针数组的前提是所有的函数类型都相同,都是两个参,参数类型都是int,返回值都是int。

六.指向函数指针数组的指针

其实本质上就是一个数组指针,只不过这个数组的类型是一个函数指针类型数组。

在这里插入图片描述

要分辨类型,需要先看与哪个操作符结合。

一些练习

在这里插入图片描述

七.回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个 函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

应用例子qsort函数,由于篇幅比较大,将其放到了下一篇博客里qsort函数

在这里插入图片描述

相关内容

热门资讯

“十五五”开好局起好步丨从“便...   福建、上海等地都在新年第一个工作日召开了优化营商环境专题大会,部署今年的新任务、新目标,优化营商...
这才是中国村味的“顶流”!到贵...   拒绝流量套路,只凭 “乡土味” 出圈!贵州的 “村字号” 为何能让全网上头?带你解锁贵州破圈密码...
专访|“中国汽车品牌的文化独特...   新华社伦敦1月16日电 专访|“中国汽车品牌的文化独特性正逐步显现”——访英国汽车产业专家贝利 ...
新春近 暖汤沸——林芝市工布江...   “刚从雪山脚下过来,泡进这温泉里,浑身的寒气一下就散了!抬头是青山,身旁是热气,这体验太绝了!”...
数读中国开局新活力 | 规模已...   编者按:2026年是“十五五”开局之年。内需市场潜力持续释放,消费新场景不断涌现,文旅融合、冰雪...
​错账更正的方法有哪些 错账更正的方法有哪些在进行任何更正之前,首先要明确的是,所有更正都应基于准确无误的原则来进行。这意味...
​流动比率的分析要点 流动比率的分析要点流动比率=流动资产/流动负债*100%或者[(期初流动资产+期末流动资产)/2]/...
百万医疗险榜:锁定20年的星相... 很多人以为,百万医疗险都差不多,反正都是报销住院费用。但现实往往是,续保问题才是隐形雷区。市面上多数...
​应付票据怎么背书转让 应付票据怎么背书转让应付票据的背书转让方式主要有以下几种:1.全称背书:即将票据背书人的全称写在票据...
普京同伊朗总统通电话   新华社消息,据俄罗斯媒体16日报道,俄罗斯总统普京同伊朗总统佩泽希齐扬通电话。