【C语言】_使用冒泡排序模拟实现qsort函数

news2025/1/12 14:47:25

目录

1. 排序函数的参数

2. 排序函数函数体

2.1 比较元素的表示

2.2 交换函数Swap的实现

2.3 排序函数bubble_sort的实现

3. 测试整型数据排序

3.1 整型数据比较函数cmp_int的实现

3.2 整型数据排序后输出函数print_int的实现

3.3 整型数据测试函数test_int的实现 

3.4 完整程序及运行结果 

4. 测试结构体型数据排序

4.1 创建结构体型数据

4.2 结构体型数据比较函数cmp_stu_byxxxx的实现

4.3 结构体型数据排序后输出函数print_stu的实现

4.4 结构体型数据测试函数test_stu的实现

4.5 完整程序及运行结果


 qsort采用快排实现,现使用冒泡进行模拟实现;

 关于排序,冒泡排序实现文章参考如下:
【C语言】_冒泡排序及其优化思路-CSDN博客icon-default.png?t=O83Ahttps://blog.csdn.net/m0_63299495/article/details/145014576

关于qsort函数,具体使用方法文章参考如下:

【C语言】_qsort函数-CSDN博客icon-default.png?t=O83Ahttps://blog.csdn.net/m0_63299495/article/details/145076745

1. 排序函数的参数

void bubble_sort(void* base, // 参数1:泛型指针接收待排序数组基址
	size_t sz,				 // 参数2:正数变量接收待排数据个数
	size_t width,            // 参数3:正数变量接收单个待排数据大小
	int(*cmp)(const void* p1,const void* p2) 
// 参数4:函数指针变量接收待排数据大小比较函数地址、
{  }

注:理解函数指针的重要作用,正是由于函数指针cmp的实现,才实现了多种类型元素的比较;

2. 排序函数函数体

2.1 比较元素的表示

1、对于冒泡排序,比较原则为相邻元素进行比较。

对于原排整型数据的冒泡排序,可使用>=<对 arr[ j ]与arr[ j+1 ]直接进行判断;

但为实现各种类型数据的排序,则需重新编写元素比较函数cmp;

2、关于相邻元素的表示,当前待排序数组基址为base,待排序元素大小为width,

对于第 j 个与第 j+1 个元素,可将base强转为char*类型后偏移对应倍数的数据元素大小width

即表示为:(char*)base + j × width (char*)base + (j+1) × width

cmp((char*)base+j*width,(char*)base+(j+1)*width)

2.2 交换函数Swap的实现

1、对于原排整型数据的冒泡排序,可创建整型临时变量tmp对arr[ j ]与arr[ j+1 ]进行交换;

但对于多种类型数据,编写时临时变量不能确定为某一具体类型,

单独封装交换函数Swap以实现交换功能;

2、关于Swap函数的参数类型,由于已强转为char*类型,故其参数类型直接写为char*类型即可;

3、对于Swap函数,仅有待交换元素的起始指针并不能完成交换,还需提供待交换元素大小

void Swap(char* buf1, char* buf2,size_t width) 
{  }

4、由于元素大小未知,可令待交换元素逐字节进行交换,交换元素大小width次

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

2.3 排序函数bubble_sort的实现

void bubble_sort(void* base, // 参数1:泛型指针接收待排序数组基址
	size_t sz,				 // 参数2:正数变量接收待排数据个数
	size_t width,            // 参数3:正数变量接收单个待排数据大小
	int(*cmp)(const void* p1,const void* p2)) {  // 参数4:函数指针变量接收待排数据大小比较函数地址
	// 确定趟数: 
	// 对于sz个数,需排sz-1趟
	int i = 0;
	for (int i = 0; i < sz - 1; i++) {
		// 1趟排序内:
		// 假设该序列已经有序:
		int flag = 1;
		int j = 0;
		// 确定1趟内比较次数:
		// 对于第i趟,待排序数个数:sz-i个,需比较的数的对数:sz-1-i对
		for (j = 0; j < sz - 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);
				// 进入循环体发生交换<=>序列非有序,将标志重置为0:
				flag = 0;
			}
		}
		// 本趟未交换,则表示序列已经有序,终止后续趟数
		if (flag == 1) {
			break;
		}
	}
}

3. 测试整型数据排序

3.1 整型数据比较函数cmp_int的实现

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

3.2 整型数据排序后输出函数print_int的实现

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

3.3 整型数据测试函数test_int的实现 

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

3.4 完整程序及运行结果 

#include<stdio.h>
int cmp_int(const void* p1, const void* p2) {
	return *(int*)p1 - *(int*)p2;
}
void Swap(char* buf1, char* buf2,size_t width) {
	for (int i = 0; i < width; i++) {
		char* tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}
void bubble_sort(void* base, // 参数1:泛型指针接收待排序数组基址
	size_t sz,				 // 参数2:正数变量接收待排数据个数
	size_t width,            // 参数3:正数变量接收单个待排数据大小
	int(*cmp)(const void* p1,const void* p2)) {  // 参数4:函数指针变量接收待排数据大小比较函数地址
	// 确定趟数: 
	// 对于sz个数,需排sz-1趟
	int i = 0;
	for (int i = 0; i < sz - 1; i++) {
		// 1趟排序内:
		// 假设该序列已经有序:
		int flag = 1;
		int j = 0;
		// 确定1趟内比较次数:
		// 对于第i趟,待排序数个数:sz-i个,需比较的数的对数:sz-1-i对
		for (j = 0; j < sz - 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);
				// 进入循环体发生交换<=>序列非有序,将标志重置为0:
				flag = 0;
			}
		}
		// 本趟未交换,则表示序列已经有序,终止后续趟数
		if (flag == 1) {
			break;
		}
	}
}
void print_int(int* arr, int sz) {
	for (int i = 0; i < sz; i++) {
		printf("%d ", *(arr + i));
	}
}
void test_int() {
	int arr[] = { 9,7,5,3,1,8,6,4,2,0 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	bubble_sort(arr,sz,sizeof(arr[0]),cmp_int);
	print_int(arr, sz);
}
int main() {
	test_int();
	return 0;
}

运行结果如下:

4. 测试结构体型数据排序

4.1 创建结构体型数据

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

4.2 结构体型数据比较函数cmp_stu_byxxxx的实现

由于结构体有多个成员变量,分别编写对应排序函数:

int cmp_stu_byname(const void* p1, const void* p2) {
	strcmp(((Stu*)p1)->name, ((Stu*)p2)->name);
	//strcmp( (*((Stu*)p1)).name, (*((Stu*)p1)).name );
}
int cmp_stu_byage(const void* p1, const void* p2) {
	return ((Stu*)p1)->age - ((Stu*)p2)->age;
	/*return (*((Stu*)p1)).age - (*((Stu*)p1)).age;*/
}

4.3 结构体型数据排序后输出函数print_stu的实现

void print_stu(Stu* arr, int sz) {
	for (int i = 0; i < sz; i++) {
		printf("name:%s, age:%d\n", arr[i].name, arr[i].age);
	}
}

4.4 结构体型数据测试函数test_stu的实现

void test_stu() {
	struct Stu arr[3] = { {"zhangsan",20},{"lisi",19},{"wangwu",21} };
	int sz = sizeof(arr) / sizeof(arr[0]);

	printf("sorted by name:\n");
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_stu_byname);
	print_stu(arr, sz);
	printf("\n");
	printf("sorted by age:\n");
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_stu_byage);
	print_stu(arr, sz);
}

4.5 完整程序及运行结果

#include<stdio.h>
#include<string.h>
typedef struct Stu {
	char name[20];
	int age;
}Stu;
int cmp_stu_byname(const void* p1, const void* p2) {
	strcmp(((Stu*)p1)->name, ((Stu*)p2)->name);
	//strcmp( (*((Stu*)p1)).name, (*((Stu*)p1)).name );
}
int cmp_stu_byage(const void* p1, const void* p2) {
	return ((Stu*)p1)->age - ((Stu*)p2)->age;
	/*return (*((Stu*)p1)).age - (*((Stu*)p1)).age;*/
}
void print_stu(Stu* arr, int sz) {
	for (int i = 0; i < sz; i++) {
		printf("name:%s, age:%d\n", arr[i].name, arr[i].age);
	}
}
void test_stu() {
	struct Stu arr[3] = { {"zhangsan",20},{"lisi",19},{"wangwu",21} };
	int sz = sizeof(arr) / sizeof(arr[0]);

	printf("sorted by name:\n");
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_stu_byname);
	print_stu(arr, sz);
	printf("\n");
	printf("sorted by age:\n");
	bubble_sort(arr, sz, sizeof(arr[0]), cmp_stu_byage);
	print_stu(arr, sz);
}
void bubble_sort(void* base, // 参数1:泛型指针接收待排序数组基址
	size_t sz,				 // 参数2:正数变量接收待排数据个数
	size_t width,            // 参数3:正数变量接收单个待排数据大小
	int(*cmp)(const void* p1,const void* p2)) {  // 参数4:函数指针变量接收待排数据大小比较函数地址
	// 确定趟数: 
	// 对于sz个数,需排sz-1趟
	int i = 0;
	for (int i = 0; i < sz - 1; i++) {
		// 1趟排序内:
		// 假设该序列已经有序:
		int flag = 1;
		int j = 0;
		// 确定1趟内比较次数:
		// 对于第i趟,待排序数个数:sz-i个,需比较的数的对数:sz-1-i对
		for (j = 0; j < sz - 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);
				// 进入循环体发生交换<=>序列非有序,将标志重置为0:
				flag = 0;
			}
		}
		// 本趟未交换,则表示序列已经有序,终止后续趟数
		if (flag == 1) {
			break;
		}
	}
}
int main() {
	test_stu();
	return 0;
}

运行结果如下:

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

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

相关文章

ECharts饼图下钻

背景 项目上需要对Echarts饼图进行功能定制&#xff0c;实现点击颜色块&#xff0c;下钻显示下一层级占比 说明 饼图实现点击下钻/面包屑返回的功能 实现 数据结构 [{name: a,value: 1,children: [...]},... ]点击下钻 // 为图表绑定点击事件&#xff08;需要在destroy…

Java聊天小程序

拟设计一个基于 Java 技术的局域网在线聊天系统,实现客户端与服务器之间的实时通信。系统分为客户端和服务器端两类,客户端用于发送和接收消息,服务器端负责接收客户端请求并处理消息。客户端通过图形界面提供用户友好的操作界面,服务器端监听多个客户端的连接并管理消息通…

蓝桥杯嵌入式速通(1)

1.工程准备 创建一文件夹存放自己的代码&#xff0c;并在mdk中include上文件夹地址 把所有自身代码的头文件都放在headfile头文件中&#xff0c;之后只需要在新的文件中引用headfile即可 headfile中先提前可加入 #include "stdio.h" #include "string.h"…

net-http-transport 引发的句柄数(协程)泄漏问题

Reference 关于 Golang 中 http.Response.Body 未读取导致连接复用问题的一点研究https://manishrjain.com/must-close-golang-http-responsehttps://www.reddit.com/r/golang/comments/13fphyz/til_go_response_body_must_be_closed_even_if_you/?rdt35002https://medium.co…

TrustRAG:增强RAG系统鲁棒性与可信度的创新框架

在人工智能飞速发展的今天&#xff0c;大语言模型&#xff08;LLMs&#xff09;凭借其强大的语言处理能力在诸多领域大放异彩。检索增强生成&#xff08;RAG&#xff09;系统&#xff08;面向企业RAG&#xff08;Retrieval Augmented Generation&#xff09;系统的多维检索框架…

业务链指标,用户行为模式识别,埋点系统

个人博客&#xff1a;无奈何杨&#xff08;wnhyang&#xff09; 个人语雀&#xff1a;wnhyang 共享语雀&#xff1a;在线知识共享 Github&#xff1a;wnhyang - Overview 此前&#xff0c;我们已经成功搭建起最为基础的聚类指标体系&#xff0c;涵盖计数、求和、最大值、最小…

Git撤销指定commit并更新远端仓库

Git撤销指定commit并更新远端仓库 一、撤销指定commit 1.首先执行git log 命令&#xff0c;查看git历史提交以及commit信息&#xff1a; 由于需要脱敏&#xff0c;所以截图可能看得马赛克比较多&#xff0c;需要关注的就是上面的commit后跟的id&#xff0c;以及HEAD当前指定…

基于DFT与IIR-FIR滤波器的音频分析与噪声处理

基于DFT与IIR-FIR滤波器的音频分析与噪声处理 【完整源码文档报告】 【需要可随时联系博主&#xff0c;常在线能秒回!】 系统功能与实现介绍 功能与实现 音频处理系统界面搭建&#xff1a;利用MATLAB的GUI工具&#xff0c;构建了音频分析界面&#xff0c;包括文件导入、录…

基于单片机的无线气象仪系统设计(论文+源码)

1系统方案设计 如图2.1所示为无线气象仪系统设计框架。系统设计采用STM32单片机作为主控制器&#xff0c;结合DHT11温湿度传感器、光敏传感器、BMP180气压传感器、PR-3000-FS-N01风速传感器实现气象环境的温度、湿度、光照、气压、风速等环境数据的检测&#xff0c;并通过OLED1…

MySQL库表的操作

目录 一、库的操作 1.1库的创建 1.2字符集和校验规则 1.2.1 查看系统默认字符集以及校验规则 1.2.2 查看数据库支持的字符集 1.2.3 查看数据库支持的字符集校验规则 1.2.4 校验规则对数据库的影响 1.3操纵数据库 1.3.1显示库 1.3.2显示创建语句 1.3.3修改数据库 1.3…

硬件设计-齐纳管

目录 摘要 详情 齐纳管的工作电流、 摘要 齐纳管&#xff08;Zener Diode&#xff09;是一种特殊的二极管&#xff0c;它能够在特定的反向电压下保持电流稳定。正常情况下&#xff0c;二极管只允许正向电流通过&#xff0c;而阻止反向电流流过。而齐纳管在一定的反向电压下可…

Airflow:TimeSensor感知时间条件

在数据管道工作流中&#xff0c;任务可能需要在特定的时间执行&#xff0c;或者在继续之前等待一定的时间。为了满足这些需求&#xff0c;Apache Airflow提供了TimeSensor&#xff0c;这是一种内置Sensor&#xff0c;可以监控当前时间&#xff0c;并在达到指定时间时触发后续任…

libusb学习——简单介绍

文章目录 libusb 简介libusb 编译libusb 源码目录介绍核心代码文件平台支持例子 API使用libusb初始化和去初始化libusb设备处理和枚举libusb 杂项libusb USB描述符libusb 设备热插拔事件通知libusb 异步设备I/Olibusb 同步设备I/Olibusb 轮询与定时 libusb 涉及技术参考 libusb…

HTML5 网站模板

HTML5 网站模板 参考 HTML5 Website Templates

阿里云ios镜像源

阿里云镜像源&#xff1a;阿里巴巴开源镜像站-OPSX镜像站-阿里云开发者社区 下载centos7

【大数据】Apache Superset:可视化开源架构

Apache Superset是什么 Apache Superset 是一个开源的现代化数据可视化和数据探索平台&#xff0c;主要用于帮助用户以交互式的方式分析和展示数据。有不少丰富的可视化组件&#xff0c;可以将数据从多种数据源&#xff08;如 SQL 数据库、数据仓库、NoSQL 数据库等&#xff0…

【2024年华为OD机试】 (A卷,100分)- 端口合并(Java JS PythonC/C++)

一、问题描述 题目描述 有 M 个端口组 (1 < M < 10)&#xff0c; 每个端口组是长度为 N 的整数数组 (1 < N < 100)&#xff0c; 如果端口组间存在 2 个及以上不同端口相同&#xff0c;则认为这 2 个端口组互相关联&#xff0c;可以合并。 输入描述 第一行输入端…

【灵码助力安全3】——利用通义灵码辅助智能合约漏洞检测的尝试

前言 随着区块链技术的快速发展&#xff0c;智能合约作为去中心化应用&#xff08;DApps&#xff09;的核心组件&#xff0c;其重要性日益凸显。然而&#xff0c;智能合约的安全问题一直是制约区块链技术广泛应用的关键因素之一。由于智能合约代码一旦部署就难以更改&#xf…

AOP实现操作日志记录

文章目录 1.common-log4j2-starter1.目录2.pom.xml 引入依赖3.LogAspect.java4.Log4j2AutoConfiguration.java Log4j2自动配置类条件注入切面类 2.common-log4j2-starter-demo 测试1.目录2.application.yml 启用日志切面3.TraceController.java4.结果 1.common-log4j2-starter …

分布式ID—雪花算法

背景 现在的服务基本是分布式、微服务形式的&#xff0c;而且大数据量也导致分库分表的产生&#xff0c;对于水平分表就需要保证表中 id 的全局唯一性。 对于 MySQL 而言&#xff0c;一个表中的主键 id 一般使用自增的方式&#xff0c;但是如果进行水平分表之后&#xff0c;多…