C语言之深入指针及qsort函数(五)(详解介绍)

C语言之深入指针

在这篇博客看不懂的可以看看这篇C语言之深入指针(四)在上篇博客中介绍了:

  1. 函数指针变量
  2. 函数指针数组
  3. 简易计算器的实现\

文章目录

  • C语言之深入指针
    • 1 回调函数
    • 2 qsort函数的使用
      • 2.1 使用冒泡排序排序整型数组
      • 2.2 使用qsort函数排序整型数组
      • 2.2 使用qsort函数排序结构体数组
        • 2.2.1 按照年龄来排序
        • 2.2.2 按照名字来排序
      • 3 qsort函数总结

1 回调函数

使用回调函数修改前的代码:

#include <stdio.h>

void menu()
{
	printf("********************************\n");
	printf("*******   1. Add   2. Sub  *****\n");
	printf("*******   3. Mul   4. Div  *****\n");
	printf("*******   0. exit         ******\n");
	printf("********************************\n");
}

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;
}
int main()

{
	int input = 0;
	int x = 0;
	int y = 0;
	int ret = 0;
	do
	{
		menu();
		printf("请选择:>>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			printf("请输入操作数:>>");
			scanf("%d %d", &x, &y);
			ret = Add(x, y);
			printf("%d\n", ret);
			break;
		case 2:
			printf("请输入操作数:>>");
			scanf("%d %d", &x, &y);
			ret = Sub(x, y);
			printf("%d\n", ret);
			break;
		case 3:
			printf("请输入操作数:>>");
			scanf("%d %d", &x, &y);
			ret = Mul(x, y);
			printf("%d\n", ret);
			break;
		case 4:
			printf("请输入操作数:>>");
			scanf("%d %d", &x, &y);
			ret = Div(x, y);
			printf("%d\n", ret);
			break;
		case 0:
			printf("退出计算器\n");
			break;
		default:
			printf("请重新输入\n");
		}
	} while (input);
	return 0;
}
case 1:
			printf("请输入操作数:>>");
			scanf("%d %d", &x, &y);
			ret = Add(x, y);
			printf("%d\n", ret);
			break;

在上述代码中,可以看到在每个case语句中,代码基本相似,基本逻辑是一致的,但是输入和输出部分完全一致这么写显得代码很冗余,那么我们可以使用回调函数来减少代码的重复

回调函数:回调函数就是⼀个通过函数指针调⽤的函数
如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被⽤来调⽤其所指向的函数时,被调⽤的函数就是回调函数。回调函数不是由该函数的实现⽅直接调⽤,⽽是在特定的事件或条件发⽣时由另外的⼀⽅调⽤的,⽤于对该事件或条件进⾏响应

使用回调函数修改后的代码:

#include <stdio.h>

void menu()
{
	printf("********************************\n");
	printf("*******   1. Add   2. Sub  *****\n");
	printf("*******   3. Mul   4. Div  *****\n");
	printf("*******   0. exit         ******\n");
	printf("********************************\n");
}

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 cacl(int (*p) (int, int) )
{
	int x = 0;
	int y = 0;
	int ret = 0;
	printf("请输入操作数:>>");
	scanf("%d %d", &x, &y);
	ret = p(x, y);
	printf("%d\n", ret);
}
int main()

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

在上述代码中,使用了回调函数 cacl ,将函数的地址传给了回调函数,在回调函数中使用了加法等函数,减少了代码的重复
cacl函数的形参部分为函数指针类型用来接收函数的地址

2 qsort函数的使用

2.1 使用冒泡排序排序整型数组

给定一个整型数组,要求将其排序
最简单的方法就是使用冒泡排序
代码如下:

#include <stdio.h>
void bobble_sort(int arr[], int sz)
{
	int i = 0;
	int tmp = 0;
	for (i = 0; i < sz-1; i++)
	{
		int j = 0;
		for (j = 0; j < sz - 1 - i; j++)
		{
			if (arr[j] > arr[j + 1])
			{
				tmp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = tmp;
			}
		}
	}
}

Print(int arr[], int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
}
int main()

{
	int arr[] = { 1,4,7,2,5,8,10,3,6,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	Print(arr, sz);
	bobble_sort(arr, sz);
	Print(arr, sz);
	return 0;
}

2.2 使用qsort函数排序整型数组

在2.1中使用冒泡排序排序了整型数组,但是bobble_sort这个函数只能排序整型数组,在C语言中有这么一个函数——qsort可以排序任意类型的数组

void qsort (void* base,
            size_t num, 
            size_t size,            
            int (*compar)(const void*,const void*));//qsort函数的声明

在这里插入图片描述
(图片转载至https://cplusplus.com/)
在cplusplus中介绍到,qsort函数有四个形参

  1. void* base   //base指向的数组中第一个的元素  void* 为泛型指针可以接收任意类型的指针
  2. size_t num  //base指向的数组中元素的个数(待排序数组中的元素个数)
  3. size_t size  //base指向的数组中元素的大小(单位是字节)
  4. int (compar)(const void,const void*)) //函数指针 - 指针指向的函数是用来比较数组中的2个元素的
  5. 4中函数指针的例子:int compar (const void* p1, const void* p2);

我将在代码中详细介绍:

#include <stdio.h>
#include <stdlib.h>    //使用qsort需要包含的头文件

//int compar (const void* p1, const void* p2);
int sort_intarr(const void* p1, const void* p2)  //比较数组中两个元素 形参类型是固定的
{
	return (*(int*)p1 - *(int*)p2);
}

Print(int arr[], int sz)  //打印输出
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
}
int main()
{
	int arr[] = { 1,4,7,2,5,8,10,3,6,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);   //计算数组元素个数
	Print(arr, sz);  //排序前打印一次
	qsort(arr,sz,sizeof(arr[0]), sort_intarr);
	Print(arr, sz); //排序前打印一次后
	return 0;
}
qsort(arr,sz,sizeof(arr[0]), sort_intarr);

一、
在上述代码中传给了qsort函数4个参数
1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 sort_intarr为自己写的一个函数,用于比较两个元素

int sort_intarr(const void* p1, const void* p2)
{
	return (*(int*)p1 - *(int*)p2);
}

二、
在上述代码中定义了一个sort_intarr函数用来比较两个元素的大小
return ( * (int * )p1 - (int * )p2);
1 由于形参为void*类型的数据,需要转换成int类型的数据进行比较

TIPS:如果想降序排列可以将return ( * (int * )p1 - * (int * )p2);改为return ( * (int * )p2 - * (int * )p1);

2.2 使用qsort函数排序结构体数组

由于结构体中的数据类型很多,所以我们得按需求来实现函数

2.2.1 按照年龄来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>

struct Stu
{
	char name[20];
	int age;
};

int cmp_struct_by_age(const void* p1, const void* p2)
{
	return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;
	//return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;
	//两段代码等价
}

int main()
{
	struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_age);
	return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_age为自己写的一个函数,用于比较两个元素

//两段代码等价
return ((struct Stu*)p1)->age - ((struct Stu*)p1)->age;
return (*(struct Stu*)p1).age - (*(struct Stu*)p1).age;

二、
1 由于是结构体数组,所以要将void*类型的数据强制转换成结构体类型
2 如果使用 ( -> ) 操作符结构体数组不使用解引用操作符来使用
3 要想使用 ( . ) 操作符的话,需要在前面加上解引用操作符

2.2.2 按照名字来排序

代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>   //使用strcmp函数需要包含的头文件

struct Stu
{
	char name[20];
	int age;
};

int cmp_struct_by_name(const void* p1, const void* p2)
{
	return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
	//return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);
	//两段代码等价
}

int main()
{
	struct Stu arr[] = { {"zhangsan",25},{"lisi",18} ,{"wangwu",30} };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), cmp_struct_by_name);
	return 0;
}

一、
与排序整型数组一样,传入的参数都是那4种

1 arr为数组名,即传给了qsort第一个元素
2 sz为计算出的元素个数
3 sizeof(arr[0])为计算出一个元素的大小
4 cmp_struct_by_name为自己写的一个函数,用于比较两个元素

//两段代码等价
return strcmp(((struct Stu*)p1)->name, ((struct Stu*)p2)->name);
return strcmp((*(struct Stu*)p1).name, (*(struct Stu*)p2).name);

二、
两个字符串之间的比较不是比较字符串长度,而是比较字符的ASCII
例如:

char str1 = "abcdef";
char str2 = "abz";

第一个字母都是a相等的话会比较下一对字符,c的ASCII值小于g的ASCII,所以str2的长度大于str1,我们可以使用strcmp函数来比较字符串的大小

3 qsort函数总结

1 qsort可以排序任意类型的数据
2 使用qsort需要包含头文件 <stdlib.h>
3 使用qsort需要传4个参数
4 根据需要排序的数据传递不同的函数 按照参数和返回值实现
5 升序使用第一个形参 - 第二个形参 降序则反之

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

web 前台页面内弹出框(一)

本文已经不推荐在使用了&#xff0c;有更加优秀的 &#xff0c;详情参考layui弹出层 前端当前页面编辑一些数据时&#xff0c;往往会用到弹出窗口&#xff0c;但每个页面单独修改有显得比较麻烦&#xff0c;因此&#xff0c;可以建立一个公用的方法&#xff0c;去掉用就可以了&…

分类预测 | Matlab实现PSO-BiLSTM-Attention粒子群算法优化双向长短期记忆神经网络融合注意力机制多特征分类预测

分类预测 | Matlab实现PSO-BiLSTM-Attention粒子群算法优化双向长短期记忆神经网络融合注意力机制多特征分类预测 目录 分类预测 | Matlab实现PSO-BiLSTM-Attention粒子群算法优化双向长短期记忆神经网络融合注意力机制多特征分类预测分类效果基本描述程序设计参考资料 分类效果…

基于STM32的循迹小车项目实战

循迹小车是一种能够沿着预定路线行驶的智能小车&#xff0c;通过巡线传感器检测路面的线路&#xff0c;并根据检测结果调整行驶方向。本项目将基于STM32微控制器实现一个简单的循迹小车&#xff0c;通过学习和实践&#xff0c;帮助初学者熟悉STM32的开发流程和掌握循迹小车的实…

IntelliJ IDEA启动一个普通的java web项目的配置

原创/朱季谦 这是我很久以前刚开始用IntelliJ IDEA时记录的笔记&#xff0c;应该是五年前的一篇笔记了。正好赶上最近离职了&#xff0c;可以有比较多的时间把以前的记录整理一下&#xff0c;可以让刚接触到IntelliJ IDEA的童鞋学习如何在IntelliJ IDEA引入一个单机版的jar形式…

flutter跨端开发for Web、Windows QA (持续补充中)

flutter跨端开发for Web、Windows Q&A Q1 开发环境运行web 解决跨域问题 问题描述 : 常见于本地调试项目 本地项目 10.125.10 如图所示 请求项目接口 解决方案&#xff1a; 开发环境运行web 解决跨域问题 flutter run -d chrome --web-browser-flag "--disable-web-s…

利用OpenCV做个熊猫表情包 二

之前写了一篇 利用OpenCV做个熊猫表情包吧_Leen的博客-CSDN博客 回想起来觉得有点太弱了&#xff0c;意犹未尽&#xff0c;每次使用需要自己去手动截取人脸&#xff0c;清除黑边什么的才能使用demo去合成表情&#xff0c;于是有空的时候就改进了一下&#xff0c;让它利用open…

SpringBoot整合Quartz示例

数据表 加不加无所谓,如果需要重启服务器后重新执行所有JOB就把sql加上 如果不加表 将application.properties中的quartz数据库配置去掉 自己执行自己的逻辑来就好,大不了每次启动之后重新加载自己的逻辑 链接&#xff1a;https://pan.baidu.com/s/1KqOPYMfI4eHcEMxt5Bmt…

除了chatGPT网站外,国内有些可以使用的AI网站 文心一言 讯飞星火 豆包 通义千问 人工智能网站 AI网站

2023年随着人工智能技术的不断发展&#xff0c;AI网站如ChatGPT等越来越受到人们的关注。这些网站具有多种作用&#xff0c;可以帮助人们更方便地获取信息、解决问题&#xff0c;甚至进行创作。 首先&#xff0c;AI网站可以提供智能问答服务。与传统的搜索引擎相比&#xff0c…

ClickHouse的表引擎

1 表引擎的使用 表引擎是ClickHouse的一大特色。可以说&#xff0c; 表引擎决定了如何存储表的数据。包括&#xff1a; 数据的存储方式和位置&#xff0c;写到哪里以及从哪里读取数据。支持哪些查询以及如何支持。并发数据访问。索引的使用&#xff08;如果存在&#xff09;。是…

Python大数据之linux学习总结——day09_hive调优

hive调优 hive官方配置url: https://cwiki.apache.org/confluence/display/Hive/ConfigurationProperties hive命令和参数配置 hive参数配置的意义: 开发Hive应用/调优时&#xff0c;不可避免地需要设定Hive的参数。设定Hive的参数可以调优HQL代码的执行效率&#xff0c;或帮…

ClickHouse SQL操作

基本上来说传统关系型数据库&#xff08;以MySQL为例&#xff09;的SQL语句&#xff0c;ClickHouse基本都支持&#xff0c;这里不会从头讲解SQL语法只介绍ClickHouse与标准SQL&#xff08;MySQL&#xff09;不一致的地方。 1 Insert 基本与标准SQL&#xff08;MySQL&#xff09…

力扣刷题篇之数与位3

系列文章目录 目录 系列文章目录 前言 数学问题 总结 前言 本系列是个人力扣刷题汇总&#xff0c;本文是数与位。刷题顺序按照[力扣刷题攻略] Re&#xff1a;从零开始的力扣刷题生活 - 力扣&#xff08;LeetCode&#xff09; 数学问题 204. 计数质数 - 力扣&#xff08;Le…

IoC和DI

Spring 是包含众多工具的 IoC 容器,存的是对象,对象这个词在 Spring 的范围内,称之为 bean IoC 是控制反转 控制权进行了反转,比如对某一个东西的控制权在 A 手上,结果变成了 B ,Spring 管理的是 bean ,所以这里的控制权指的是 bean 的控制权,也就是对象的控制权进行了反转 …

DNS1(Bind软件)

名词解释 1、DNS&#xff08;Domain Name System&#xff09; DNS即域名系统&#xff0c;它是一个分层的分布式数据库&#xff0c;存储着IP地址与主机名的映射 2、域和域名 域为一个标签&#xff0c;而有多个标签域构成的称为域名。例如hostname.example.com&#xff0c;其…

海康Visionmaster-环境配置:VB.Net 二次开发环境配 置方法

Visual Basic 进行 VM 二次开发的环境配置分为三步。 第一步&#xff0c;使用 VS 新建一个框架为.NET Framework 4.6.1&#xff0c;平台去勾选首选 32 为的工程&#xff0c;重新生成解决方案&#xff0c;保证工程 Debug 下存在 exe 文件&#xff0c;最后关闭新建工程&#xff1…

python趣味编程-5分钟实现一个简单贪吃蛇游戏(含源码、步骤讲解)

在本教程《如何用 Python 制作游戏》中,我们将为名为“简单贪吃蛇游戏”的游戏制作一个程序。 Python 中的贪吃蛇游戏:项目信息 项目名称:Python 游戏摘要:Python提供了一个名为pygame的内置库,用于开发游戏。使用的语言:Python 与 Tkinter GUI 库Python版本(推荐):2.…

yolov5模型代码怎么修改

yaml配置文件 深度乘积因子 宽度乘积因子 所有版本只有这两个参数的不同&#xff0c;s m l x逐渐加宽加深 各种类型层参数对照 backbone里的各层&#xff0c;在这里解析&#xff0c;只需要改.yaml里的各层参数就能控制网络结构 修改网络结构 第一步&#xff1a;把新加的模块…

Educational Codeforces Round 20 A-E

文章目录 A. Maximal Binary MatrixB. Distances to ZeroC. Maximal GCDD. Magazine AdE. Roma and Poker A. Maximal Binary Matrix 思路&#xff1a;一道很有意思的构造&#xff0c;我们可以发现&#xff0c;按照下述&#xff0c;从外到内进行一层一层的构造一定是最优的。 …

系列六、JVM的内存结构【栈】

一、产生背景 由于跨平台性的设计&#xff0c;Java的指令都是根据栈来设计的&#xff0c;不同平台的CPU架构不同&#xff0c;所以不能设计为基于寄存器的。 二、概述 栈也叫栈内存&#xff0c;主管Java程序的运行&#xff0c;是在线程创建时创建&#xff0c;线程销毁时销毁&…

48v变12v同步转换芯片

48v变12v同步转换芯片 以下是一篇关于48V变12V同步转换器WD5105ic的文章正文&#xff1a;48V变12V同步转换器WD5105ic是一种电源管理芯片&#xff0c;它可以将48V的直流电压转换为12V的直流电压。这款芯片具有广泛的应用范围&#xff0c;包括车载充电器件、电动车仪表器件、电…