您现在的位置是:首页 > 编程 > 

第六节——从深层剖析qsort的使用(让你不再害怕指针)

2025-07-19 01:25:49
第六节——从深层剖析qsort的使用(让你不再害怕指针) 1.什么是回调函数回调函数就是⼀个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另⼀个函数,当这个指针被用来调用其所指向的函数时,被调用的函数就是回调函数。回调函数不是由该函数的实现⽅直接调用,而是在特定的事件或条 件发生时由另外的⼀方调用的,用于对该事件或条件进行响应。看完后,也许不是很通透,下面用这个代码简单解释下:

第六节——从深层剖析qsort的使用(让你不再害怕指针)

1.什么是回调函数

回调函数就是⼀个通过函数指针调用的函数。

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

看完后,也许不是很通透,下面用这个代码简单解释下:

回顾上节使用的转移表,我们发现用switch来实现过于冗余,而明白了回调函数的原理后能大大简化代码。

可参考这篇文章:第五节——转移表(让你不再害怕指针)


2.qsort的使用

前言:冒泡排序内容

学完冒泡排序后,我们发现只能排序整形数据,那么要排字符数据,结构数据等等时,又是否能交给冒泡排序解决呢? 很明显,冒泡排序并不能完美地比较这些数据。那么,有没有一个函数能够实现这些数据的排序呢?

答案是肯定的,通过使用qsort可以实现任何类型的排序。

C++最权威

上面两段的含义

qsort排序整形数据

了解完qsort的思想及使用后,便可以使用qsort排序整形数据。

代码语言:javascript代码运行次数:0运行复制
#include<stdio.h>
#include<stdlib.h>

int com_int(ct void* e1, ct void* e2)
{
	return *(int*)e1 - *(int*)e2;
}

void print(int arr[], int sz)
{
	int i = 0;
	for (;i < sz;i++)
	{
		printf("%d ", arr[i]);
	}
}

void test1()
{
	int arr[10] = { 2,,5,6,82,4,6,67,10,0 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	qsort(arr, sz, sizeof(arr[0]), com_int);
	print(arr, sz);
}

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

思路理解:

  1. 先把arr数组的首元素的地址传给qsort函数,
  2. 接着告诉qsort函数有多少数据(即sz),
  3. 知道个数后,qsort并不知道一次要跳过多少个字节,因此需要告诉告诉qsort这个arr数组一个元素的长度(即字节)
  4. 最后按照qsort的使用实现两个函数的比较。在com_int函数中,发明者并不知道客户传过来的是什么类型的数据,因此设为void型指针(可以接受任何类型的数据),最后进行强制类型转换即可。

qsort排序结构体数据

代码语言:javascript代码运行次数:0运行复制
#include<stdio.h>
#include<stdlib.h>

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

int cmp_struct(ct void* e1, ct void* e2)
{
	return (*(struct Stu*)e1).age - (*(struct Stu*)e2).age;
}


void test2()
{
	struct Stu s[] = { {"zhangsan",15},{"lisi",18},{"wangwu",20} };
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), cmp_struct);
}

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

qsort排序字符串数据

排序字符串数据时,我们会遇到一个问题:a和b如何进行比较呢? 这时要使用strcmp函数(用来比较字符串大小)。

代码语言:javascript代码运行次数:0运行复制
//strcmp和qsort返回值一样
//strcmp(字符串1,字符串2)
//如果字符串1 大于 字符串2,返回>0的数字
//如果字符串1 等于 字符串2,返回0
//如果字符串1 小于 字符串2,返回<0的数字

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

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

int cmp_struct(ct void* e1, ct void* e2)
{
	return strcmp((*(struct Stu*)e1).name) - strcmp((*(struct Stu*)e2).name);
}


void test2()
{
	struct Stu s[] = { {"zhangsan",15},{"lisi",18},{"wangwu",20} };
	int sz = sizeof(s) / sizeof(s[0]);
	qsort(s, sz, sizeof(s[0]), cmp_struct);
}

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

总的来说,qsort可以接收任何类型的数据,并且完成两两数据的大小交换。


.qsort的模拟实现

那么我们能不能实现一个函数,就像qsort一样能排序任意类型的数据 1.qsort是按照快速排序的思想。 2.在冒泡排序的基础上,改造成能够排序任意类型的数据。

详解冒泡排序原理

qsort的模拟实现

代码语言:javascript代码运行次数:0运行复制
#include<stdio.h>

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

void Swap(char* buf1, char* buf2, size_t width)
{
	int i = 0;
	for (i = 0;i < width;i++)
	{
		char tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}

int cmp_int(ct void* e1, ct void* e2)
{
	return *(int*)e1 - *(int*)e2;
}

void bubble_sort(void* base, size_t num, size_t width, int (*cmp)(ct void* e1, ct void* e2))
{
	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 * width, (char*)base + (j + 1) * width) > 0)
			{
				Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
			}
		}
	}
}

void test()
{
	int arr[10] = { 0,1,2,,4,5,6,8,7,9 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_int);
	print(arr, sz);
}

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

我们来逐层分析这段代码的实现:

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。 原始发表:2024-12-14,如有侵权请联系 cloudcommunity@tencent 删除数据指针字符串函数排序

#感谢您对电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格的认可,转载请说明来源于"电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格

本文地址:http://www.dnpztj.cn/biancheng/1133919.html

相关标签:无
上传时间: 2025-07-18 23:29:16
留言与评论(共有 19 条评论)
本站网友 效应
8分钟前 发表
qsort排序结构体数据代码语言:javascript代码运行次数:0运行复制#include<stdio.h> #include<stdlib.h> struct Stu { char name[20]; int age; }; int cmp_struct(ct void* e1
本站网友 醋泡姜的功效
2分钟前 发表
7
本站网友 城市发展规划
3分钟前 发表
7
本站网友 魔法卡片登陆
18分钟前 发表
int sz) { int i = 0; for (i = 0;i < sz;i++) { printf("%d "
本站网友 唐镇二手房
13分钟前 发表
0 }; int sz = sizeof(arr) / sizeof(arr[0]); qsort(arr
本站网友 误判心理学
19分钟前 发表
字符串2) //如果字符串1 大于 字符串2
本站网友 郑州交通银行
7分钟前 发表
18}
本站网友 安诺优达
19分钟前 发表
因此设为void型指针(可以接受任何类型的数据)
本站网友 苍井空是什么人
14分钟前 发表
qsort排序结构体数据代码语言:javascript代码运行次数:0运行复制#include<stdio.h> #include<stdlib.h> struct Stu { char name[20]; int age; }; int cmp_struct(ct void* e1
本站网友 试管全过程
5分钟前 发表
sizeof(s[0])
本站网友 股票图形技术分析
5分钟前 发表
qsort排序结构体数据代码语言:javascript代码运行次数:0运行复制#include<stdio.h> #include<stdlib.h> struct Stu { char name[20]; int age; }; int cmp_struct(ct void* e1
本站网友 红豆汤
10分钟前 发表
6
本站网友 porttunnel
14分钟前 发表
sizeof(s[0])
本站网友 44511
22分钟前 发表
char* buf2
本站网友 雅莹服饰
17分钟前 发表
cmp_struct); } int main() { test2(); return 0; }总的来说
本站网友 邝字怎么读
5分钟前 发表
便可以使用qsort排序整形数据
本站网友 北部万科城
9分钟前 发表
15}
本站网友 燕都紫庭
8分钟前 发表
sz); } int main() { test1(); return 0; }思路理解:先把arr数组的首元素的地址传给qsort函数