编写基于冒泡排序算法的qsort函数

news2024/11/18 3:39:35

目录

1.简单认识冒泡排序

 2.进入正文分析如何实现函数

3.1比较两个相邻元素的大小

3.2比较两个相邻元素大小后要换函数

4.my_qsort函数:

5.总结:


1.简单认识冒泡排序

冒泡排序的步骤如下:

  • 比较相邻的两个元素,如果第一个元素比第二个元素大(或小),就交换它们的位置。
  • 对每一对相邻的元素重复上述操作,直到数组的末尾。这样,最大(或最小)的元素就被移动到了数组的最后一个位置。
  • 除了最后一个元素外,对剩余的元素重复以上步骤,直到没有任何一对相邻元素需要交换为止
// 冒泡排序
void bubble_sort(int arr[], int len)
{
	int i, j, temp;
	for (i = 0; i < len - 1; i++)
	{
		for (j = 0; j < len - 1 - i; j++)
		{
			if (arr[j] > arr[j + 1])
			{
				temp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = temp;
			}
		}
	}
}

 我们可以基于冒泡排序思想对以上代码进行改造成qsort函数对任意数据类型起到排序作用

了解qsort函数可以看下面的网站或者我的另一篇博客

cplusplus网站:https://legacy.cplusplus.com/reference/cstdlib/qsort/?kw=qsort

关于qsort函数的使用博客:qsort库函数的使用_Jamo@的博客-CSDN博客

 2.进入正文分析如何实现函数

我们通过观察冒泡函数可以看出我们在最初给的冒泡函数中只能比较整型数据来进行排序

 在冒泡函数中的第二个for循环中我们通过每一趟冒泡来依次比较两个相邻的元素大小来决定是否交换他们的位置,但如果我们在想要排序的数组中遇到了浮点型数据呢?又或是字符型数据呢?又或是结构体类型数据呢?

显然此时的冒泡函数无法解决我们的燃眉之急。

但我们依然可以借助于冒泡的模板来写出自己的qsort函数来解决问题:

  • 我们发现在排序时我们只是排序的数据类型不一样了,但排序思想任然是冒泡思想,因此我们做出的改变就是对第二个for循环中的比较方法就行改进。

3.1比较两个相邻元素的大小

对于int 类型数据我们可以通过大于小于来直接比较他们的大小来决定是否交换位置

对于所有类型来说我们可以实现一个比较函数来帮我们解决这个问题:

//在比较两个相邻元素大小时,由于不知道跳过元素有多大,因此在处理不确定数据类型排序时,使用char * 类型和原数据类型size大小来找到冒泡排序的下两对元素


compar((char*)数据1 , (char*)数据2 )

//基于冒泡排序算法的qsort函数
void bubble_qsort(void* base, size_t num, size_t size, int (*compar)(const void* e1, const void* e2))
{
	int i = 0;
	int j = 0;
	int tmp = 0;
	for (i = 0; i < num - 1; i++)
	{
		//为了处理不同数据类型比较方法,此处的排序需要在原来整型数据冒泡排序写法上进行改造
		for (j = 0; j < num - 1 - i; j++)
		{



            //在为比较函数compar找两两元素时,由于不知道跳过元素有多大,因此在处理不确定数据类型排序时,使用char * 类型和原数据类型size大小来找到冒泡排序的下两对元素

            //比如说比较元素是int类型时(char*)base加上循环变量j * int类型大小4找到数组首个元素地址,第二个元素地址便是(cahr*)base加上循环变量(j+1)之后 *4 找到第二个元素地址
 
			if (compar((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				//交换
				swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
			}
		}
	}
}

 以比较整型数据举例:我们自己使用qsort函数时写出自己想要比较的数据类型的compar函数:

//比较函数
//返回大于0的数字代表前一个元素大于后一个元素
//返回等于0的数字代表前一个元素等于后一个元素
//返回小于0的数字代表前一个元素小于后一个元素
int compar(const void* e1, const void* e2)
{
	return *(int*)e1 - *(int*)e2;
}

3.2比较两个相邻元素大小后要换函数

我们通过自己写一个swap函数来解决这一问题;

为什么要自己写交换函数而不用第三个变量来进行两两交换呢?

我们既然是为了写出一个qsort函数比较任意数据类型数据那我们自然也不知道我们将来要交换的元素究竟是什么类型的,自然也无法创建第三个变量来使其两两交换

实现swap函数:

//给swap函数初始化参数与交换函数compar同理
swap((char*)base + j * size, (char*)base + (j + 1) * size, size);

//由于不知道交换元素的类型,因此我们决定对相邻两个元素一个字节一个字节进行交换
//将两元素
void swap(char* p1, char* p2, size_t size)
{
	int i = 0;
	for (i = 0; i < size; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}
}

4.my_qsort函数:

//此交换函数原理是对内存中相邻元素一个字节一个字节交换
void swap(char* p1, char* p2, size_t size)
{
	int i = 0;
	for (i = 0; i < size; i++)
	{
		char tmp = *p1;
		*p1 = *p2;
		*p2 = tmp;
		p1++;
		p2++;
	}

}
//比较函数
//返回大于0的数字代表前一个元素大于后一个元素
//返回等于0的数字代表前一个元素等于后一个元素
//返回小于0的数字代表前一个元素小于后一个元素
int compar(const void* e1, const void* e2)
{
	return *(int*)e1 - *(int*)e2;
}


//基于冒泡排序算法的qsort函数
void bubble_qsort(void* base, size_t num, size_t size, int (*compar)(const void* e1, const void* e2))
{
	int i = 0;
	int j = 0;
	int tmp = 0;
	for (i = 0; i < num - 1; i++)
	{
		//为了处理不同数据类型比较方法,此处的排序需要在原来整型数据冒泡排序写法上进行改造
		for (j = 0; j < num - 1 - i; j++)
		{
			//不知道跳过元素有多大,因此在处理不确定数据类型排序时,使用char * 类型和原数据类型size大小来找到冒泡排序的下两对元素

			//返回大于0的数字代表前一个元素大于后一个元素
			//返回等于0的数字代表前一个元素等于后一个元素
			//返回小于0的数字代表前一个元素小于后一个元素
			if (compar((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			{
				//交换
				swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
			}
		}
	}
}
int main()
{
	int num_arr[10] = { 10,9,8,7,6,5,4,3,2,1 };
	int sz = sizeof(num_arr) / sizeof(num_arr[0]);
	printf("原来的顺序:");
	print_arr(num_arr, sz);

	bubble_qsort(num_arr, sz, sizeof(num_arr[0]), compar);

	printf("排序的顺序:");
	print_arr(num_arr, sz);

	return 0;
}

5.总结:

实现该函数最主要的部分便是交换函数compar函数参数的书写,如何在不知道元素数据类型的情况下找到元素来进行大小比较,以及如何在不知道元素数据类型的情况下对两个相邻元素来交换。


以上便是全部内容了,感谢大家的支持和鼓励,下次见!

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

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

相关文章

TS编译选项——不允许使用隐式any类型、不明确类型的this、严格检查空值、编译后文件自动设置严格模式

一、不允许使用隐式any类型 在tsconfig.js文件中配置noImplicitAny属性 {"compilerOptions": {// 不允许使用隐式any类型"noImplicitAny": true} } 开启后即可禁止使用隐式的any类型 注意&#xff1a;显式的any类型并不会被禁止 二、不允许使用不明确类…

亚马逊儿童自行车,滑板车等电动移动设备合规标准UL报告如何办理?UL 2272、UL 2849

加拿大 儿童自行车 儿童自行车适用于 14 岁以下儿童。儿童自行车的车轮由两个轮子组成&#xff0c;一个在另一个后面&#xff0c;通过踩踏推动&#xff0c;用连接在前轮上的车把操纵。其中一些可能配备有训练轮&#xff0c;这是一对平行于后轮的额外的车轮&#xff0c;可防止自…

ShapeableImageView 不只是圆形ImageView

偶然间看到了这位老哥的 https://juejin.cn/post/6869376452040196109#comment 文章&#xff0c;发现了ShapeableImageView–一个多形状的ImageView &#xff0c;虽然似乎发布了很久了&#xff0c;现在学习不晚。 效果图 布局文件 <com.google.android.material.imageview.S…

yyyy/MM/dd与yyyy-MM-dd使用new Date().getTime()时间转换不相等?!!

起源 该问题发现于日期组件增加国家法定假节假日的禁用&#xff0c;通过pickerOptions属性定义disabledDate方法实现 export default {data () {return {holidayList: [2023-01-01,2023-01-02,2023-01-21,2023-01-22,2023-01-23,2023-01-24,2023-01-25,2023-01-26,2023-01-27…

SQL注入——预编译CASE注入

文章目录 预编译 CASE 注入1. SQL注入漏洞防御2. WEBGOAT SQL注入2.1 WebGoat 8.02.2 Order by 注入2.2.1 构造 when 的条件2.2.2 代码审计 预编译 CASE 注入 预编译 CASE&#xff08;Prepared CASE&#xff09;是一种数据库查询语言&#xff08;如SQL&#xff09;中的控制语句…

Android Studio 的android.jar文件在哪儿

一般在&#xff1a;C:\Users\admin\AppData\Local\Android\Sdk\platforms\android-33下&#xff08;不一定是33&#xff0c;这个得看你Android Studio->app->builde.gradle的targetSdk是多少&#xff09; 怎么找&#xff1a; 1.打开Android Studio 粘贴地址后&#xff0…

【C++】stack queue

stack & queue 一、容器适配器二、deque&#xff08;了解&#xff09;三、stack1. stack 的介绍2. 模拟实现 stack 四、queue1. queue 的使用2. 模拟实现 queue3. priority_queue&#xff08;1&#xff09;priority_queue 的介绍&#xff08;2&#xff09;priority_queue 的…

数字IC笔试千题解--单选题篇(二)

前言 出笔试题汇总&#xff0c;是为了总结秋招可能遇到的问题&#xff0c;做题不是目的&#xff0c;在做题的过程中发现自己的漏洞&#xff0c;巩固基础才是目的。 所有题目结果和解释由笔者给出&#xff0c;答案主观性较强&#xff0c;若有错误欢迎评论区指出&#xff0c;资料…

如何通过Python实现接口自动化的参数关联?

前言 通常在接口自动化中&#xff0c;经常会参数关联的问题&#xff0c;那么什么是参数关联&#xff1f; 参数关联就是上一个接口的返回值会被下一个接口当做参数运用&#xff0c;其中Python中可以实现参数关联的方法有很多种&#xff0c;今天小编给大家介绍下&#xff0c;如…

IC验证| Verilog语法详解之条件语句

Verilog 是一种用于数字逻辑电路设计的硬件描述语言&#xff0c;可以用来进行数字电路的仿真验证、时序分析、逻辑综合。 既是一种行为级&#xff08;可用于电路的功能描述&#xff09;描述语言又是一种结构性&#xff08;可用于元器件及其之间的连接&#xff09;描述语言。 …

RedisTemplate出现\xac\xed\x00\x05t\x00\x0f前缀解决

问题描叙 出现这种乱码前缀的原因是没有进行序列化&#xff0c;因此导致在传输过程出现乱码问题&#xff0c;存到数据库&#xff0c;发现 key,hash key/value 都有 \xAC\xED\x00\x05t\x00 前缀。RedisTemplate类中默认是没有设置序列化的。 解决方法 设置RedisTemplate的序列…

如何快速建立一个专业、高效的宠物医院小程序?

随着社会的发展和科技的进步&#xff0c;人们对于宠物的关注度越来越高&#xff0c;养宠物已经成为了许多人的生活方式。然而&#xff0c;宠物的健康问题也随之而来&#xff0c;宠物医院成为了不可或缺的存在。为了更好地服务于宠物主人&#xff0c;打造一个专属的宠物医院线上…

IntelliJ IDEA 左侧Commit栏不见了

1.点击File->Settings->Version Control->Commit 2.勾选Use non-modal commit interface

关于ElementUI之首页导航与左侧菜单实现

目录 一.Mock 1.1.什么是Mock.js 1.2.特点 1.3.安装与配置 1.3.1. 安装mock.js 1.3.2.引入mock.js 1.4.mockjs使用 1.4.1.定义测试数据文件 1.4.2.mock拦截Ajax请求 1.4.3.界面代码优化 二.总线 2.1.是什么 2.2.前期准备 2.3.配置组件与路由关系 2.3.1. 配置组件 …

Mock.js之Element-ui搭建首页导航与左侧菜单

&#x1f3ac; 艳艳耶✌️&#xff1a;个人主页 &#x1f525; 个人专栏 &#xff1a;《Spring与Mybatis集成整合》《springMvc使用》 ⛺️ 生活的理想&#xff0c;为了不断更新自己 ! 1、Mock.js的使用 1.1.什么是Mock.js Mock.js是一个模拟数据的生成器&#xff0c;用来帮助前…

希尔排序代码及时间空间复杂度

希尔排序&#xff08;Shell Sort&#xff09;是一种插入排序的改进算法&#xff0c;它通过将数据分成多个小组来排序&#xff0c;然后逐渐减小这些小组的间隔&#xff0c;直到最后一次使用标准的插入排序算法。希尔排序的时间复杂度取决于使用的间隔序列&#xff0c;通常为 O(n…

Vulnhub-driftingbules:5 靶机复现完整过程

记录对driftingbules:5 靶机的复现过程 kali的IP地址&#xff1a;192.168.200.14 靶机IP地址&#xff1a;192.168.200.60 一、信息收集 1.对利用nmap目标靶机进行扫描 由于arp-scan属于轻量级扫描&#xff0c;在此直接使用nmap进行对目标靶机扫描开放端口 nmap -A -p 1-65…

从SmartPay dll学到的内容 宏定义 单件模式 迭代 日志记录函数进入与出来

日志记录函数进入与出来&#xff1a;利用C的反初始化来记录退出 函数运行记时、调用次数统计等 宏定义 配置里的宏 WIN32;NDEBUG;_WINDOWS;_USRDLL;SMARTPAY_PGLDLL_192787_EXPORTS;ESLOG_RELEASE;HAVE_STRUCT_TIMESPEC;%(PreprocessorDefinitions)减少代码耦合 关闭日志等 …

国内音视频开发的前景怎么样?

国内音视频开发的前景怎么样? 本人就是音视频开发&#xff0c;谈一下我的观点。 目前干我们这一行的年纪都比较大&#xff0c;我自己工作五年就是很年轻的了。年会上老板说除了音视频中心的大家都是比较年轻的。。。 有些也是过了35岁了&#xff0c;四十的都有。是不是觉得这…