【C语言】指针进阶(2)

news2024/11/24 4:00:13

接上期文章指针进阶(1)指针进阶(1)

目录

1.函数指针

2.函数指针数组

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

4.回调函数

4.1qsort的用法 

 void*类型的指针介绍

 使用qsort对数组进行排序

 使用qsort对结构体进行排序:

4.2使用冒泡排序算法模拟实现qsort


1.函数指针

        我们已经学过了数组指针,而数组指针是指向数组的指针。根据类比的思想,我们可以推测出函数指针是指向函数的指针。

        那么,如何才能得到函数的地址呢?请看下面的代码:

 程序输出的两个地址是相同的:

 要想得到函数地址,可以使用&函数名或者直接函数名,使用%p的格式打印出来就可以了。

由此我们可知:

函数名是函数的地址

&函数名也是函数地址

如果想要将上面代码中函数的地址保存到指针变量中,那么这个指针的类型该如何书写呢?

 测验:写出下面函数的函数指针类型

 答案是:

 注意:*pf的括号不能去掉,必须使*先和pf结合。

通过函数指针来调用该函数

下面代码是函数指针的调用方法:

int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int(*pf)(int, int) = Add;
	int r = Add(3, 5);
	printf("%d\n", r);

	int m = (*pf)(3, 5);
	printf("%d\n", m);

	int n = pf(3, 5);
	printf("%d\n", n);

	return 0;
}

打印结果:

 通过上面的代码我们可以发现,pf前面的*可以省略,实际上,pf前面的*只是一个摆设,无论你加多少个*,都不会影响结果:

阅读两段有趣的代码

代码一:

分析代码:

 当然,我们写的程序是不能直接访问0地址的。

代码二:

 分析代码:

2.函数指针数组

        在前面,我们已经知道了将整型指针存放在一个数组中,那么这个数组就叫做整型指针数组,像:

int*arr[5] 就是一个整形指针数组。当然,还有我们比较熟悉的字符指针数组 char*arr[5] .

由此,我们可以推断出,函数指针数组就是数组的每个元素是函数指针类型

        为了学习函数指针数组,我们用一个实现两个整形数字计算的程序说起:

        假设有这样一个计算器,它要实现两个整数的加法,减法,乘法,除法运算:

 首先,我用写了下面这几个实现上述功能的函数:

int Add(int x, int y)
{
	return x + y;
}
int Sub(int x, int y)
{
	return x - y;
}
int Mul(int x, int y)
{
	return x * y;
}
int Div(int x, int y)
{
	return x / y;
}

可以看出,这四个函数的返回类型,参数个数,参数类型都是一样的,像这种同类型的函数就可以将函数地址存放在函数指针数组中。

关键是,这个函数指针数组该怎么写呢?我们一步一步的来写:

先写出能指向这四个函数的函数指针:

要想得到函数指针数组,只需要在上面代码的基础上修改:

得到数指针数组只需要在函数指针名字后面加上一个 [ ] 即可。

函数指针数组实际应用场景:
仍然以上面的计算器为例,下面是使用普通方法来写一个简单的计算器程序:

#include<stdio.h>

int Add(int x, int y)
{
	return x + y;
}
int Sub(int x, int y)
{
	return x - y;
}
int Mul(int x, int y)
{
	return x * y;
}
int Div(int x, int y)
{
	return x / y;
}

void meun()
{
	printf("***********************\n");
	printf(" 1:add           2:sub \n");
	printf(" 3:mul           4:div \n");
	printf(" 0:exit                \n");
	printf("***********************\n");
}

int main()
{
	int input = 0;
	int x = 0, y = 0;
	int ret = 0;
	do
	{
		meun();
		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;
}

使用普通方法来写代码,在switch语句中有非常多的重复部分。

下面对main函数进行修改(用到函数指针数组的方式):

int main()
{
	int input = 0;
	int x = 0, y = 0;
	int ret = 0;
	//函数指针数组
	int (*pfArr[5])(int, int) = { NULL,Add,Sub,Mul,Div };
	//下标                         0    1   2   3   4

	do
	{
		meun();
		printf("请选择:>");
		scanf("%d", &input);
		if (input >= 1 && input <= 4)
		{
			printf("请输入两个操作数:>");
			scanf("%d %d", &x, &y);
			ret = pfArr[input](x, y);//调用相应的函数
			printf("ret = %d\n", ret);
		}
		else if(input==0)
		{
			printf("退出计算器\n");
		}
		else
		{
			printf("选择错误,重新选择\n");
		}
	} while (input);
	return 0;
}

 

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

指向函数指针数组的指针这部分内容我们只需要了解即可,不需要深入研究。

在上面计算器的例子中,我们已经知道了函数指针数组的写法:

现在,我们想要写一个指针来指向这个函数指针数组,该怎么写呢?

只需要将数组名pfArr换成指针名p,在p的前面再加上一个*,并用括号括起来([ ]的结合性比*高)

4.回调函数

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

 就拿上面写过的计算器来举例子,我们把main()函数再次修改:

void calc(int(*pf)(int, int))
{
	int x = 0, y = 0;
	int ret = 0;
	printf("请输入两个操作数:>");
	scanf("%d %d", &x, &y);
	ret = pf(x, y);
	printf("ret = %d\n", ret);
}

int main()
{
	int input = 0;
	
	do
	{
		meun();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			calc(Add);
			break;
		case 2:
			calc(Sub);
			break;
		case 3:
			calc(Mul);
			break;
		case 4:
			calc(Div);
			break;
		case 0:
			printf("退出计算器\n");
			break;
		default:
			printf("选择错误,重新选择:>");
			break;
		}
	} while (input);

	return 0;
}

我们多增加了下面这个函数,这个函数的形参是一个函数指针,用来接收Add,Sub, Mul, Div等函数。当调用calc这个函数时,calc中又会根据形参传过来的函数地址调用这个函数。这就是函数回调。

 程序的执行过程:

 

4.1qsort的用法 

在标准库中,有一个函数叫qsort,是用来排序的,qsort适合于任意类型的数据排序。

qsort的使用说明

 compar的返回值所表示的内容:

 void*类型的指针介绍

void*的指针是无具体类型的指针,这种指针是不能直接解引用的,也不能直接进行运算。但它可以接收任意类型的参数地址,使用前要进行强制类型转换。

 void*指针接收任意类型的参数地址

 

 使用qsort对数组进行排序

#include<stdio.h>
#include<stdlib.h>
int cmp_int(const void* p1, const void* p2)
{
	return *(int*)p1 - *(int*)p2;
}
void Print(int arr[], int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
}
void test1()
{
	int arr[10] = { 9,8,7,6,5,4,3,2,1,0 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), cmp_int);
	Print(arr, sz);
}

int main()
{
	test1();
	return 0;
}

 使用qsort对结构体进行排序:

需要通过调试来观察数组的变化

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
//qsort排序整形数组
struct stu
{
	char name[20];
	int age;
};
int cmp_stu_by_age(const void* p1, const void* p2)
{
	return ((struct stu*)p1)->age - ((struct stu*)p2)->age;
}
int cmp_stu_by_name(const void* p1, const void* p2)
{
	return strcmp(((struct stu*)p1)->name, ((struct stu*)p2)->name);
}
void test2()
{
	struct stu arr[] = { {"zhangsan",20},{"lisi",50},{"wangwu",15} };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), cmp_stu_by_name);
}
int main()
{
	test2();
	return 0;
}


4.2使用冒泡排序算法模拟实现qsort

 由于没有想学习快速排序算法,所以我们使用冒泡排序的思想,实现一个功能类似qsort的函数。

要点:

1.使用冒泡排序算法的思想

2.适用于任意数据类型的排序

要解决实现对任意类型数据的排序,我们接收数组的指针应该使用void*类型。

当我们排序整形时,我们可以直接进行比较,但当我们如果要进行结构体大小比较时,就不能直接比较了,这里需要用户自己写一个比较函数来进行比较。在排序算法中以函数参数的形式传递用户写的比较函数。

在整形数组中,可以直接通过数组下标来访问数组元素,但在我们自己写的排序函数中,我们使用的void*的指针类型,无法通过下标来访问。我们可以用下面的方式得到数组每个元素的地址:

 不同的数据类型交换略有差异,原因是在排序函数中不知道该数据的类型,同时也是使用void*的指针接收的地址。但是,我们是知道每个元素的字节大小的,我们只需要一字节一字节交换,交换size次,就可以实现交换功能了:

 代码实现交换:

接下来看看完整的实现代码吧:

//交换
void swap(char* buf1, char* buf2, int size)
{
	int i = 0;
	for (i = 0; i < size; i++)
	{
		char tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}

void bubble_sort(void *base,int num,int size,int(*cmp)(const void*, const void *))
{
	int i = 0;
	for (i = 0; i < num - 1; i++)
	{
		int j = 0;
		for (j = 0; j < num - 1 - i; j++)
		{
			if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				//交换
				swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
			}
		}
	}
}

这篇博客就到这里结束了,下篇博客将继续学习指针,下篇博客是指针的最后一篇了,将会有大量有关指针面试题的总结,欢迎大家阅读与点赞~~

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++】-初步认识和学习继承

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

【CAS6.6源码解析】在IDEA中调试可插拔的supprot模块

CAS源码的casWebApplication启动后&#xff0c;默认只加载最小的支撑系统的模块&#xff0c;很多模块&#xff08;大部分在support包下&#xff09;是需要手动去引入的&#xff08;对新人来说有坑&#xff09;&#xff0c;这里介绍一下如何手动引入这些模块。 文章目录 调试步骤…

TCP通信 -- 文件传输

www 客户端&#xff1a; package com.itheima.b03TCPTest3;import java.io.*; import java.net.InetAddress; import java.net.Socket; import java.net.UnknownHostException;public class Client {public static void main(String[] args) throws IOException {System.out.p…

HTTP超本文传输协议

HTTP超本文传输协议 HTTP简介HTTP请求与响应HTTP请求请求行请求头空行请求体 HTTP响应响应行响应头空行响应体 HTTP请求方法GET和POST之间的区别HTTP为什么是无状态的cookie原理session 原理cookie 和 session 的区别cookie如何设置cookie被禁止后如何使用session HTTP简介 HT…

idea快速运行vue项目

目录 一、前提 二、步骤 安装vue.js插件 添加脚本 进行如下配置 一、前提 安装好node.js环境并初始化完成和安装好依赖 二、步骤 安装vue.js插件 打开idea,然后在File–Settings–Plugins–Makerplace下找到vue.js插件,安装并重启idea 添加脚本 进行如下配置 在Sctipts中根…

【手撕排序算法】---基数排序

个人主页&#xff1a;平行线也会相交 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 平行线也会相交 原创 收录于专栏【数据结构初阶&#xff08;C实现&#xff09;】 我们直到一般的排序都是通过关键字的比较和移动这两种操作来进行排序的。 而今天介绍的…

MAC 推送证书不受信任

配置推送证书的时候&#xff0c;一打开就变成不受信任&#xff0c;搜了很多解决版本。 由于苹果修改相关规定&#xff0c;推送证书 打开Apple PKI - Apple 下载AppleWWDRCA文件&#xff0c;选择G4,双击安装之后&#xff0c;证书已经变为受信任。 AppleWWDRCA(Apple Worldwid…

一、大数据技术之Flume(简介)

第1章 Flume概述 1.1 Flume定义 Flume是Cloudera提供的一个高可用的&#xff0c;高可靠的&#xff0c;分布式的海量日志采集、聚合和传输的系统。Flume基于流式架构&#xff0c;灵活简单。 1.2 Flume基础架构 Flume组成架构如下图所示。 1.2.1 Agent Agent是一个JVM进程&…

每日一题——反转链表

题目 给定一个单链表的头结点pHead(该头节点是有值的&#xff0c;比如在下图&#xff0c;它的val是1)&#xff0c;长度为n&#xff0c;反转该链表后&#xff0c;返回新链表的表头。 数据范围&#xff1a;0≤n≤1000 要求&#xff1a;空间复杂度 O(1) &#xff0c;时间复杂度 O…

C# Modbus通信从入门到精通(21)——Modbus TCP协议原理

Modbus TCP是走网口的&#xff0c;也可以在同一时间内有多个从站访问主站&#xff0c;并且通过Modbus事务处理标识来区分同一时刻的不同Modbus事务&#xff0c;这是区别于Modbus ASCII和Modbus RTU的地方。 1、访问模式&#xff1a; Modbus客户端通常输入Modbus服务器的IP地址…

科技与人元宇宙论坛跨界对话

近来&#xff0c;“元宇宙”成为热门话题&#xff0c;越来越频繁地出现在人们的视野里。大家都在谈论它&#xff0c;但似 乎还没有一个被所有人认同的定义。元宇宙究竟是什么&#xff1f;未来它会对我们的工作和生活带来什么样 的改变&#xff1f;当谈论虚拟现实&#xff08;VR…

代码随想录| 图论04 查并集 ●查并集理论知识 ●1971.寻找图中是否存在路径 ●684.冗余连接 ●685.冗余连接II

#查并集理论知识 并查集用处&#xff1a;解决连通性问题 将两个元素添加到一个集合中。判断两个元素在不在同一个集合 思路&#xff1a;将三个元素A&#xff0c;B&#xff0c;C &#xff08;分别是数字&#xff09;放在同一个集合&#xff0c;其实就是将三个元素连通在一起&a…

【Kubernetes资源篇】ingress-nginx最佳实践详解

文章目录 一、Ingress Controller理论知识1、Ingress Controller、Ingress简介2、四层代理与七层代理的区别3、Ingress Controller中封装Nginx&#xff0c;为什么不直接用Nginx呢&#xff1f;4、Ingress Controller代理K8S内部Pod流程 二、实践&#xff1a;部署Ingress Control…

Raki的读paper小记:LORA: LOW-RANK ADAPTATION OF LARGE LANGUAGE MODELS

Abstract&Introduction&Related Work 研究任务 对大模型进行部分微调 已有方法和相关工作 现有技术通常通过扩展模型深度引入推理延迟&#xff08;Houlsby 等人&#xff0c;2019&#xff1b;Rebuffi 等人&#xff0c;2017&#xff09;&#xff0c;或通过减少模型可用序…

Python实现HBA混合蝙蝠智能算法优化卷积神经网络回归模型(CNN回归算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 蝙蝠算法是2010年杨教授基于群体智能提出的启发式搜索算法&#xff0c;是一种搜索全局最优解的有效方法…

集成了32位Cortex®M0内核XMC1302T038X0200AB、XMC1302Q040X0200AB 32MHz 200KB 闪存 工业MCU

XMC1000 32位工业 MCU 将 ARM Cortex™-M0 核心与领先的 65nm 制造工艺相结合&#xff0c;克服了目前 8 位设计的局限。XMC1000系列让目前的 8 位用户有机会享受 32 位的功耗&#xff0c;同时不在价格或易用性上做出妥协。XMC1000 在其细分市场提供最为广泛的闪存产品线&#x…

人工智能相关笔记

这近一年来&#xff0c;我在国科大修了&#xff1a;人工智能导论、机器学习与模式识别、语义网络与知识图谱、深度学习、强化学习……这几门专业课&#xff0c;由于发现了它们彼此之间有重复的知识点&#xff0c;真想把他们融会贯通一下&#xff0c;至少写个提纲挈领的东西给自…

xcode15启动IOS远程调试

1.用数据线连接IPhone到macOS 2.打开xcode15,然后点击Window->Devices and Simulators 3.选中左边的Devices可看到已连接的IPhone,然后点击Connect via network使其选中. 选择后,左边的IPhone设备的右边出现一个地球图标,表示成功通过网络连接到IPhone 现在可断开数据线的…

K8S初级入门系列之八-网络

一、前言 本章节我们将了解K8S的相关网络概念&#xff0c;包括K8S的网络通讯原理&#xff0c;以及Service以及相关的概念&#xff0c;包括Endpoint&#xff0c;EndpointSlice&#xff0c;Headless service&#xff0c;Ingress等。 二、网络通讯原理和实现 同一K8S集群&…

大模型开发(十):Chat Completion Models API 详解

全文共8000余字&#xff0c;预计阅读时间约18~28分钟 | 满满干货(附代码案例)&#xff0c;建议收藏&#xff01; 本文目标&#xff1a;详解Chat Completion Models的参数及应用实例&#xff0c;并基于该API实现一个本地知识库的多轮对话智能助理 代码&文件下载点这里 一、…