C语言易错知识点十(指针(the final))

❀❀❀ 文章由@不准备秃的大伟原创 ❀❀❀

♪♪♪ 若有转载,请联系博主哦~ ♪♪♪

❤❤❤ 致力学好编程的宝藏博主,代码兴国!❤❤❤

        许久不见,甚是想念,真的是时间时间,你慢些吧,不能再让头发变秃了 ,我愿用我一生换我头发常青~~(^▽^ )。好的好的,今天是12月31号,是2023年的最后一天,博主这里提前祝大家新年快乐!

9c86da63099546dea3b0e2bba7e19a07.gif

        最后一天了, 博主在这里问各位铁汁们一句:“2022年定下的目标2023年都实现了吗?2023又要为2024定什么目标呢?” 或许有些铁汁会骄傲的拍拍胸脯说:“何止实现了,我还超了呢!”,或许也有铁汁会说:“嗐,2023年因为这样那样的原因,没有能完成去年定的目标,很后悔!”

        其实无论你是处在哪种阶段,博主只想说:“向前看吧,过去的已经不存在了,过去一年里你是荣华富贵或是灰头土脸又如何,难道以后就不会改变了吗?就像我们头上的头发,终有一天也会变白,脱落,但是生活不还得继续吗?所以,抛下顾虑,放下过去的荣辱,在新的一年里继续努力,为更好的未来而前进吧!

         鸡汤虽多,但不喝也无用。所以,为了更好的未来,我们铁汁们还是需要继续学习的是吧,还会继续给大伟点赞收藏,继续支持大伟的对吧~对吧~对吧~ヾ(≧▽≦*)o

        呃哼,我们今天来看看C语言的核心:指针的最后一篇。今天我们要学的知识是:没了。

        8ee9a4163de74335adec0d01f38448fc.png

        啊不是,应该是有关指针的内容基本没了,这一篇主要是一些扩展,别误解哦~  好的,今天要学的内容是:回调函数,qsort使用和模拟实现,最后是有关指针的笔试题。

        诶,但是在开始之前,我们先谈一谈上一篇博客最后提到的两段代码,什么,不记得了?哼,就知道会发生这种情况,快去看—————>指针(part three)

        一.

(*(void (*)())0)();

       大家整体看这段代码是不是难于理解?那如果我们把它分为几个区块,从内部一个一个分析呢?:

void (*)()

  这是一个函数指针的声明。它声明了一个指向不带参数(void)并返回 void 的函数的指针

(void (*)())0

   这是将整数 0 转换为指定类型的函数指针

(*(void (*)())0)

   这对函数指针进行了解引用。它将地址为 0 的值视为函数并调用它

         简单来说,这段代码试图调用内存地址为 0 处的函数

        怎么说,铁汁们,从内部一步一步分解是不是就会相对好理解了?里面的内容其实是我们已经学过的知识的集合体啊,对不对。

        二.那么根据以上的思路,兄弟们自己再试图理解理解下面这个代码:

 void (*signal(int , void(*)(int)))(int);

  ——————————————————————————————————————————

signal

这是一个函数名

int

 这是signal函数的第一个参数,表示信号编号

void(*)(int)

 这是signal函数的第二个参数,是一个函数指针。它表示指向一个接受int参数并返回void的函数的指针。这个函数指针是信号处理函数,指定了在接收到信号时应采取的操作

(*signal(int, void(*)(int)))

 这部分表示signal是一个接受两个参数并返回某种结果的函数。声明的最外层指定了返回的类型

void(*)(int)

 这指定了signal函数的返回类型。它是一个函数指针,指向一个接受int参数并返回void的函数。这与signal返回信号处理函数的函数指针的概念一致

         综合起来,signal函数接受两个参数:一个表示信号编号的int,一个表示信号处理函数的函数指针。它返回一个类型为void (*)(int)的函数指针

        OK,这里把上一篇博客遗留下的问题解决了,现在开始这次的内容:

                回调函数

        定义:回调函数就是⼀个通过函数指针调⽤的函数

简单来说:就是在函数内定义了另一个函数 ,如:

int add(int a, int b)
{return a + b;
}
void calc(int(*pf)(int, int))
{int ret = 0;int x, y;printf("输⼊操作数:");scanf("%d %d", &x, &y);ret = pf(x, y);printf("ret = %d\n", ret);
}
int main()
{
calc(add);
return 0;
}

        calc函数接受一个函数指针 pf,该指针指向一个接受两个整数参数并返回整数的函数。

               qsort使用及实现

        以下是qsort函数的使用演示:

#include <stdio.h>
//qosrt函数的使⽤者得实现⼀个⽐较函数
int int_cmp(const void * p1, const void * p2)
{return (*( int *)p1 - *(int *) p2);
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}return 0;
}

        以下是cpp官方给的qsort的解释:ea6524a5d25b4d25a942e4658316bf5e.png         其中的四个元素:cf4a231fb83545b6a30949a9cd98eaee.png

        还是那句话,虽然我们程序猿不需要多好的英语,但是我们以后是需要看各种英文的文献的,所以我们从现在开始就要逐渐适应读英文的文章。

        简单来说:qsort函数内部第一个元素:要排序的数组名,第二个元素:数组的大小,第三个元素:数组内存储的类型的字节大小,第四个元素:一个cmp函数,函数内实现的是两个指针的对比。(如上,*a -*b为升序,*b - *a 为降序,*a > *b 为升序,*a < *b 为降序)

        拓展一下:qsort库函数底层是快排,时间复杂度是O(N*logN) ,在我们刷题的时候还蛮经常用的。

        那现在我们来对qsort函数进行个模拟实现,当然,是以冒泡的方法,所以时间复杂度是O(N*N)。那我们来看看吧!:

int int_cmp(const void * p1, const void * p2)
{return (*( int *)p1 - *(int *) p2);
}
void _swap(void *p1, void * p2, int size)//交换函数
{int i = 0;for (i = 0; i< size; i++){char tmp = *((char *)p1 + i);*(( char *)p1 + i) = *((char *) p2 + i);*(( char *)p2 + i) = tmp;}
}
void bubble(void *base, int count , int size, int(*cmp )(void *, void *))//这里函数嵌套函数
{int i = 0;int j = 0;for (i = 0; i< count - 1; i++){for (j = 0; j<count-i-1; j++){if (cmp ((char *) base + j*size , (char *)base + (j + 1)*size) > 0)//若前面比后面大{_swap(( char *)base + j*size, (char *)base + (j + 1)*size, size);//交换}}}
}
int main()
{int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };int i = 0;bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++){printf("%d ", arr[i]);}printf("\n");return 0;
}

        这个模拟实现是完全按照官方给的元素来写的,浓度比较高,大家好好吸收哦~

                   指针笔试题

        其实的话到这里有关指针的内容已经可以结束了,但是呢,学而不思则罔,学会了,还需要大量的练习,正如前段时间网上比较火的:9f9270086a3544da8920f0e845e01aee.png        

        题目是做不完的,但是知识点是固定的,博主这里给大家出三道题,大家珍惜哦~

        一. 

int main()
{int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int *ptr1 = (int *)(&aa + 1);int *ptr2 = (int *)(*(aa + 1));printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));return 0;
}

        解析:&aa+1越过了整个二维数组aa,所以ptr1指向的是二维数组后面的那个元素,然后ptr-1,指向的是10;而aa+1越过了整个二维数组的第一行,ptr2指向的是6,ptr2-1指向的是aa第一行的最后一个元素5。

        答案就是10和5

        二.

int main()
{char *a[] = {"work","at","alibaba"};char**pa = a;pa++;printf("%s\n", *pa);return 0;
}

        定义了一个指针数组*a[],又定义了一个二级pa指向a的首元素“work”,pa++也就是a向后走一个元素,此时*pa就是“at”。

        答案就是at

        三.

int main()
{char *c[] = {"ENTER","NEW","POINT","FIRST"};char**cp[] = {c+3,c+2,c+1,c};char***cpp = cp;printf("%s\n", **++cpp);printf("%s\n", *--*++cpp+3);printf("%s\n", *cpp[-2]+3);printf("%s\n", cpp[-1][-1]+1);return 0;
}

        同二,二级指针cp内存放的是将*c内元素逆转过来的,而三级指针指向的就是cp首元素,所以**++cpp就是指向的cpp向后走一个元素,即c+2,即“POINT”;*--*++cpp+3就是++cpp 指向 cp 数组的下一个元素,*--*++cpp 得到一个指向 c[0] 的指针 "ENTER",然后 +3 得到 "ER" ;*cpp[-2]+3 中 cpp[-2] 得到 cp[0],*cpp[-2] 得到 c[3],然后 +3 得到 "ST";cpp[-1][-1]+中,cpp[-1] 即*(cpp-1)得到 cp[2],cpp[-1][-1] 即 *(*(cpp-1)-1) 得到 c[1],然后 +1 得到 "EW"。答案就是POINT  ER  ST  EW

                

                C语言指针也就到此为止了,最后再次祝大家新年快乐,愿大家成为自己心目中的那个人。插一嘴哈~虽然指针结束了,但大伟的学习和博客还没有结束哦,希望大家以后继续支持大伟,加油!

衣带渐宽终不悔,为伊(知识)消得人憔悴    ------柳永

        本篇博客也就到此为止了,送大家一碗鸡汤,勉励自己以及这世界上所有追逐梦想的赤子趁年华尚好努力提升自己,莫欺少年穷!  6d1efb5acd724ec2897b21852c87ed4d.jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/318592.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SkyWalking 快速入门

SkyWalking 是一个基于 Java 开发的分布式系统的应用程序性能监视工具&#xff0c;专为微服务、云原生架构和基于容器&#xff08;Docker、K8s、Mesos&#xff09;架构而设计。 一、SkyWalking 简介 SkyWalking 是观察性分析平台和应用性能管理系统。 提供分布式追踪、服务网格…

MySQL是如何保证数据一致性的?

文章目录 前言MySQL保证的一致性MySQL发生不一致环节并发冲突redolog不完整binlog&redolog不一致 MySQL解决不一致方案加锁解决并发冲突undolog解决redolog不完整XA两阶段提交解决binlog和redolog的不一致 总结 前言 通过上文《MySQL是如何保证数据不丢失的&#xff1f;》…

Flink实时电商数仓之Doris框架(七)

Doris框架 大规模并行处理的分析型数据库产品。使用场景&#xff1a;一般先将原始数据经过清洗过滤转换后&#xff0c;再导入doris中使用。主要实现的功能有&#xff1a; 实时看板 面向企业内部分析师和管理者的报表面向用户或者客户的高并发报表分析 即席查询统一数仓构建&a…

gRCP - 面向未来的第二代 RPC 技术,解析 HTTP2.0 和 Protobuf

目录 一、gRCP - 面向未来的第二代 RPC 技术 1.1、gRPC 简介 1.1.1、gRPC 是个啥&#xff1f; 1.1.2、gRPC 核心设计思路 1.1.3、gRPC 和 ThriftRPC 区别 1.1.4、为什么使用 gRPC&#xff1f;&#xff08;好处&#xff09; 1.2、HTTP2.0 协议 1.2.1、回顾 HTTP1.0 和 H…

自动化测试面试题及答案大全(上)

selenium中如何判断元素是否存在&#xff1f; 没有提供原生的方法判断元素是否存在&#xff0c;一般我们可以通过定位元素异常捕获的方式判断selenium中hidden或者是display &#xff1d; none的元素是否可以定位到&#xff1f; 不可以&#xff0c;想点击的话&#xff0c;可以用…

java数据结构与算法刷题-----LeetCode746. 使用最小花费爬楼梯

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 很多人觉得动态规划很难&#xff0c;但它就是固定套路而已。其实动态规划只…

【C语言】函数

函数是什么&#xff1f; “函数”是我们早些年在学习数学的过程中常见的概念&#xff0c;简单回顾一下&#xff1a;比如下图中&#xff0c;你给函数 f(x)2*x3 一个具体的x,这个函数通过一系列的计算来返回给你一个结果(图示如下)。 这就是数学中函数的基本过程和作用。但是你…

AD教程 (二十四)PCB走线

AD教程 &#xff08;二十四&#xff09;PCB走线 信号层进行走线&#xff0c;对于信号走线尽量少打孔 电源层进行平面分割&#xff0c;电源层主要考虑载流&#xff0c;不太考虑信号的干扰 顶层不方便走线的可以在顶层打孔到底层走线&#xff0c;但是能少打孔就少打孔 交互式总…

因遭受“重大 DDoS 攻击”,官网及服务一度封停或崩溃

1月2日最新消息&#xff0c;Atomicals Market在社交媒体上发文表示&#xff0c;正遭受DDoS攻击&#xff0c;如果网站显示已被封锁&#xff0c;用户请勿担心。团队正在努力解决该问题&#xff0c;预计很快将会恢复正常。 开源建模软件 Blender 11月底也曾多次遭受严重 DDoS 攻击…

成为比开发硬气的测试人,我都经历了什么?

我的职业生涯很简单&#xff0c;可以说&#xff0c;我的测试生涯就是我的职业生涯。 大学的专业是计算机&#xff0c;当年是热门的学科&#xff0c;但自己的计算机知识不强悍&#xff0c;又加上学校的硬核是金融业&#xff0c;来学校校招的都是各大银行&#xff0c;且都是需要…

【Matlab】PSO-BP 基于粒子群算法优化BP神经网络的数据时序预测(附代码)

资源下载&#xff1a; https://download.csdn.net/download/vvoennvv/88689096 目录 【Matlab】BP 神经网络时序预测算法 【Matlab】CNN卷积神经网络时序预测算法 【Matlab】ELM极限学习机时序预测算法 【Matlab】基于遗传算法优化BP神经网络 (GA-BP)的数据时序预测 【Mat…

单机游戏数据自动保存方案

引言 单机游戏数据的自动保存方案 大家好&#xff0c;2023年还有最后的3天&#xff01; 有小伙伴私信我&#xff0c;说: 总感觉一股脑的全盘定时保存不科学&#xff0c;也写过保存变化的玩家数据&#xff0c;但是改完数据就得手动标记一下字段变化&#xff0c;感觉不够智能&…