c---冒泡排序模拟qsort

news2024/11/26 11:36:10

一、冒泡排序

二、冒泡排序优化排各种类型数据

文章目录

  • 一、冒泡排序
    • 二、冒泡排序优化排各种类型数据


冒泡排序

冒泡排序原理:两两相邻元素进行比较
在这里插入图片描述
在这里插入图片描述

初级版

void bulle_sort(int* a, int sz)
{
	int i = 0;
	for (int i = 0; i < sz-1; i++)
	{
		int j = 0; 
		for (j = 0; j < sz - 1 - i; j++)
		{
			if (a[j] > a[j+1])
			{
				int tmp = a[j];
				a[j] = a[j + 1];
				a[j + 1] = tmp;

			}
		}
	}
}

这是冒泡排序初级版,不管其原内容是否有序都会进行比较,如果原内容原本就是有序的,再每个都进行比较效率就会低下,那么这时候可以改进一下,想一个标记变量来记录是否有序,如int
falg = 0; 如果无序的情况下falg会变为1,有序的情况下falg保持0不变,如果一趟下来falg 为0
不变,那么就是有序的就不用再比较后面趟数了,这样使其在有序的情况下时间复杂度为O(n),大大提高了效率

改进版

void bulle_sort(int* a, int sz)
{
	int i = 0;
	int falg = 0;
	for (int i = 0; i < sz-1; i++)
	{
		int j = 0; 
		for (j = 0; j < sz - 1 - i; j++)
		{
			if (a[j] > a[j+1])
			{
				int tmp = a[j];
				a[j] = a[j + 1];
				a[j + 1] = tmp;
				falg = 1;
			}
		}

		if (falg == 0)
		{
			break;
		}
	}
}

冒泡排序优化排各种类型数据
上面冒泡排序可以发现只能够排序整形
在这里插入图片描述
那要是我们想利用冒泡来排其他不同类型应该如何实现呢?这里就引入c语言里的一个库函数qsort(),在cplusplus上搜索qosrt

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

可以发现这是一个排序函数,且qsort函数有四个参数,void * base目标数组,待排序的起始地址,size_t num待排序数组大小,size_t表示无符号类型,由于数组大小不可能为负数,因此设置为size_t更为合适,size_t size,数组中每个元素是多少字节,其实就是每个元素是什么类型
在这里插入图片描述

int (*compar)(const void*,const void*)

这是一个函数指针,是比较函数的函数指针,而comper实现的是比较功能,
在这里插入图片描述
比较函数在这里插入图片描述

由于比较类型不知道是什么类型的,因此用void*,这里这个设计十分合理,void*,void*存的是要比较两个元素的地址,是因为设计者在设计时不知道我们要比较什么类型的,因为void*指针可以接收任意类型变量的地址。comper函数返回类型为in类型,第一个比第二个于返回1,相等返回0,小于返回-1


qsort函数运用

int comper(const void* s1, const void* s2)
{
	return *((int*)s1) - *((int*)s2);//由于我们自己使用时知道了是什么类型,因此强转为该类型就可,
	//然后再对其解引用就可以相互进行比较了
}
int main()
{
	int arr[] = { 9,8,7,6,5,4,3,2,1 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	//bulle_sort(arr, sz);
	qsort(arr, sz, sizeof(arr[0]), comper);
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
	return 0;
}

在这里插入图片描述

可以发现出了警告是qsort未定义,这是因为没有包含它所需的头文件

在这里插入图片描述

可以往下翻找到它该用什么头文件
以qsort排序结构体

#include<string.h>
typedef struct Stu
{
	char name[20];
	int age;
}Stu;
int comper_stu_by_name(const void* s1, const void* s2)
{
	//按照名字比较,两个字符串比较是不能直接相减,用库函数strcmp进行比较
	//强制类型转换为结构体指针,然后再->找到结构体成员变量name
	return strcmp(((Stu*)s1)->name , ((Stu*)s2)->name);//由于我们自己使用时知道了是什么类型,因此强转为该类型就可,
	//得到其地址再对其解引用就可以相互进行比较了
}

int main()
{
	Stu s[3] = { {"zhangsan",20},{"wangwu",30},{"lisi",50} };
	qsort(s, sizeof(s)/sizeof(s[0]), sizeof(s[0]), comper_stu_by_name);
	for (int i = 0; i < sizeof(s)/sizeof(s[0]); i++)
	{
		printf("%s %d\n", s[i].name, s[i].age);
	}
}

在这里插入图片描述
strcmp比较字符串函数
在这里插入图片描述
在这里插入图片描述

strcmp返回类型为int
qsort可以实现任意类型的数据的排序;

以冒泡模拟qsort

//比较时需要比较什么类型自己可以定义,然后强转
//需要排不同类型只需要在这里更改就可以了
int comper(void* s1, void* s2)
{
	return *((int*)s1) - *((int*)s2);
}

void swap(char* buf1, char* buf2, int width)
{
	int i = 0;
	//width是数组中每个元素的字节大小,其实以我们来看,知道width就可以知道是什么类型,
	for (i = 0; i < width; i++)
	{
		//将每个字节都交换
		char tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}

//与qsort函数内部一致
//比较类型不明确,所有void*
void bulle_qsort( void* a, size_t sz, size_t width, int (*comper)(const void* s1, const void* s2))
{
	size_t i = 0;
	int falg = 0;
	for (int i = 0; i < sz-1; i++)
	{
		size_t j = 0; 
		for (j = 0; j < sz - 1 - i; j++)
		{
			if (comper((char*)a+j*width,(char*)a+(j+1)*width)>0)//实现比较,交换,且由于不知道要比较什么类型,,那么我们只有使用偏移量比较
			{
				swap((char*)a + j * width, (char*)a + (j + 1) * width, width);//由于不知道类型,那么就交换每个字节,把每个元素大小传过去
				falg = 1;
			}
		}

		if (falg == 0)
		{
			break;
		}
	}
}

int main()
{
	int a[] = { 9,8,7,6,5,4,3,2,1 };
	int sz = sizeof(a) / sizeof(a[0]);
	//这里可以排任意类型的数据,我这里以整形数组模拟
	
	bulle_qsort(a, sizeof(a) / sizeof(a[0]), sizeof(a[0]), comper);
	for (int i = 0; i < sz; i++)
	{
		printf("%d ", a[i]);
	}
	return 0;
}

在这里插入图片描述
冒泡模拟实现qsort就到这里了,有兴趣的小伙伴可以区试试其他类型的排序吧

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

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

相关文章

阿里大佬翻遍全网Java面试文章,总结出这份1658页文档,GitHub收获25K+点赞

就目前大环境来看&#xff0c;跳槽成功的难度比往年高很多。一个明显的感受&#xff1a;今年的面试&#xff0c;无论一面还是二面&#xff0c;都很考验Java程序员的技术功底。这不又到了面试跳槽的黄金段&#xff0c;成功升职加薪&#xff0c;不成功饱受打击。当然也要注意&…

使用日历丰富产品的用户体验

前言 经过一段时间的梳理和遴选&#xff0c;我挑选出了Android知识图谱中重要的部分&#xff0c;制作了一张脑图。读者朋友们可按照脑图查漏补缺了&#xff0c; 图片尺寸较大&#xff0c;仅附链接 。 当然&#xff0c;这是我按照自己的判断、结合参考其他博主的观点进行的挑选…

sort函数对结构体|pair对组|vector容器|map排序|二维数组的第x列 的排序

目录 sort对 vector容器 sort对 vector<pair<int,int>>对组 sort对 结构体 结构体外部规定排序 结构体内部运算符重载 map容器的排序 map的键排序 map的值排序 sort对二维数组的排序 sort对 vector容器 sort()函数可以用于对vector容器进行排序。具体来…

java基础学习 day49(JDK8的接口新特性,JDK9的新特性,接口的应用场景)

JDK8以后&#xff0c;接口中新增的默认方法 允许在接口中定义默认方法&#xff0c;需要使用关键字default修饰&#xff0c;作用为&#xff0c;解决接口升级时&#xff0c;需要强制修改所有实现类的问题接口中默认方法的定义格式&#xff1a; a. 格式&#xff1a; public defau…

【黄河流域公安院校网络空间安全技能挑战赛】部分wp

文章目录webbabyPHPfunnyPHPEzphp**遍历文件目录的类**1、DirectoryIterator&#xff1a;2、FilesystemIterator:3、**Globlterator**读取文件内容的类&#xff1a;SplFileObjectMisc套娃web babyPHP <?php highlight_file(__FILE__); error_reporting(0);$num $_GET[nu…

排序模型:DIN、DINE、DSIN

目录 DIN 输入 输出&#xff1a; 与transformer注意力机制的区别与联系&#xff1a; DINE 改善DIN 输入&#xff1a; DSIN 动机&#xff1a; DIN 适用与精排&#xff0c;论文&#xff1a; Deep Interest Network for Click-Through Rate Prediction DIN模型提出的动…

蓝桥web基础知识学习

HTMLCSS 知识点重要指数HTML 基础标签&#x1f31f;&#x1f31f;&#x1f31f;&#x1f31f;&#x1f31f;HTML5 新特性&#x1f31f;&#x1f31f;&#x1f31f;&#x1f31f;&#x1f31f;HTML5 本地存储&#x1f31f;&#x1f31f;&#x1f31f;&#x1f31f;CSS 基础语法…

Java中方法引用(引用静态方法、引用成员方法(引用其他类的成员方法、引用本类的成员方法、引用父类的成员方法)、引用构造方法、其他调用方式、小练习)

方法引用&#xff1a;把已经存在的方法拿过来用&#xff0c;当作函数式接口中抽象方法的方法体 我们前面学到Arrays工具类中的sort方法&#xff0c;当我们需要指定排序规则时&#xff0c;需要传递Comparator接口的实现类对象&#xff0c;我们之前使用匿名内部类类的形式作为参…

下一个元宇宙热点?探讨元宇宙婚礼的未来趋势

欢迎来到Hubbleverse &#x1f30d; 关注我们 关注宇宙新鲜事 &#x1f4cc; 预计阅读时长&#xff1a;7分钟 本文仅代表作者个人观点&#xff0c;不代表平台意见&#xff0c;不构成投资建议。 专家认为&#xff0c;在不久的将来&#xff0c;传统婚礼和元宇宙婚礼有可能共存…

taobao.item.barcode.update( 更新商品条形码信息 )

&#xffe5;免费必须用户授权 通过该接口&#xff0c;将商品以及SKU上得条形码信息补全 公共参数 请求地址: HTTP地址 http://gw.api.taobao.com/router/rest 公共请求参数: 公共响应参数: 请求参数 响应参数 点击获取key和secret 请求示例 TaobaoClient client new Def…

【Spring】资源操作管理:Resource、ResourceLoader、ResourceLoaderAware;

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 资源操作&#xff1a;Spring Resources一、Res…

智能家居创意产品一Homkit智能通断器

智能通断器&#xff0c;也叫开关模块&#xff0c;可以非常方便地接入家中原有开关、插座、灯具、电器的线路中&#xff0c;通过手机App或者语音即可控制电路通断&#xff0c;轻松实现原有家居设备的智能化改造。 随着智能家居概念的普及&#xff0c;越来越多的人想将自己的家改…

Pytest自动化测试框架-权威教程06-使用Marks标记测试用例

使用Marks标记测试用例通过使用pytest.mark你可以轻松地在测试用例上设置元数据。例如, 一些常用的内置标记&#xff1a;skip - 始终跳过该测试用例skipif - 遇到特定情况跳过该测试用例xfail - 遇到特定情况,产生一个“期望失败”输出parametrize - 在同一个测试用例上运行多次…

自旋锁,读写锁以及他们的异同

自旋锁 自旋锁是一种用于多线程编程的同步机制。它是一种基于忙等待的锁&#xff0c;当线程尝试获取锁时&#xff0c;如果锁已被其他线程占用&#xff0c;则该线程会一直循环检查锁是否被释放&#xff0c;直到获取到锁为止。 在自旋锁的实现中&#xff0c;使用了CPU的硬件特性…

ArcGIS制图之阴影效果的表达与运用

一、运用制图表达进行投影表达 在专题图的制作过程中&#xff0c;经常需要将目标区域从底图中进行突显&#xff0c;运用制图表达制作图层投影可以较好地实现这一目的。具体步骤如下&#xff1a; 1.将目标图层存储于数据库中并加载至窗口&#xff08;shapefile格式数据无法支持…

Android Looper简介

本文基于安卓11。 Looper是一个用具&#xff0c;在安卓程序中&#xff0c;UI线程是由事件驱动的&#xff08;onCreate, onResume, onDestory&#xff09;&#xff0c;Looper就是处理这些事件的工具&#xff0c;事件被描述为消息&#xff08;Message&#xff09;&#xff0c;Lo…

【PHP代码注入】PHP代码注入漏洞

漏洞原理RCE为两种漏洞的缩写&#xff0c;分别为Remote Command/Code Execute&#xff0c;远程命令/代码执行PHP代码注入也叫PHP代码执行(Code Execute)(Web方面)&#xff0c;是指应用程序过滤不严&#xff0c;用户可以通过HTTP请求将代码注入到应用中执行。代码注入(代码执行)…

python甜橙歌曲音乐网站平台源码

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;python音乐 获取完整源码源文件说明文档配置教程等 在虚拟环境下输入命令“python manage.py runserver”启动项目&#xff0c;启动成功后&#xff0c;访问“http://127.0.0.1:5000”进入甜橙音乐网首页&#xff0c;如图1所…

YOLOS调试记录

YOLOS是由华中科大提出的将Transformer迁移到计算机视觉领域的目标检测方法&#xff0c;其直接魔改ViT&#xff01;本文首次证明&#xff0c;通过将一系列固定大小的非重叠图像块作为输入&#xff0c;可以以纯序列到序列的方式实现2D目标检测。 模型结构 下面来调试一下该项目…

【微信小程序】-- 页面事件 - 上拉触底 - 案例(二十七)

&#x1f48c; 所属专栏&#xff1a;【微信小程序开发教程】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &…