C语言实现多种快速排序

news2025/1/11 2:22:02

目录

1.概念

2.快速排序hoare版本

2.1基本思想

2.2解释相遇处的值为何一定小于key 

2.3hoare版本快速排序的实现

3.快速排序挖坑法 

3.1基本思想

3.2挖坑法快速排序的实现

4. 快速排序前后指针版本

4.1基本思想

4.2快速排序前后指针版本实现 

 5.快速排序非递归版本

5.1基本思想

5.2快速排序非递归版本实现

6.快速排序的优化

6.1三数取中

6.1.2三数取中代码实现

6.2小区间优化

6.3优化后的代码

7.快速排序的特性总结


1.概念

        快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素序列中的某元素(一般为最左边的元素)作为基准值(key),按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止

        接下来介绍三种最基本的快速排序版本、非递归版本以及两种对快速排序的优化。

2.快速排序hoare版本

2.1基本思想

        选择最左边的元素作为key,用一个left指针指向最左边,用一个right指针指向最右边。right从最右边开始往左边走,找到比key小的元素保持不动,left从最左边开始往右边走,找到比key大的元素保持不动,然后交换left和right指向的元素,再重复上述过程,直到left和right相遇,将相遇处(相遇处的值一定小于key)的值和key交换,这样就可以将小于key的值放到左边,大于key的值放到右边,然后左右子序列重复该过程,直到所以元素都排列在相应位置上为止。

2.2解释相遇处的值为何一定小于key 

        left和right相遇只有两种情况:(1)left遇上right。(2)right遇上left。

情况1:left遇上right

        如果是该情况,表面right是保持不动的,就说明right现在找到的这个元素是小于key的,然后当left遇到right时,则是将right指向的这个元素和key交换。

情况2:right遇上left

        如果是right遇上left,是right在走动,left已经在上一轮完成了交换,现在的left指向的元素,是上一轮right找到的小于key的元素,当right遇上left时,则是将right指向的这个小于key的元素和key交换。

        注:如果确定左边的值为key,一定要从右边先开始找,不然就会出错。

2.3hoare版本快速排序的实现

void QuickSort1(int* a, int left, int right)
{
	if (left >= right)
	{
		return;
	}
	//单趟
	int begin = left; 
	int end = right;
	int keyi = left;
	while (begin < end)
	{
		while (begin < end && a[end] >= a[keyi])    //=防止找到与key相同的值进入死循环
		{
			end--;
		}
		while (begin < end && a[begin] <= a[keyi])
		{
			begin++;
		}
		Swap(&a[begin], &a[end]);
	}
	Swap(&a[keyi], &a[end]);

	keyi = end;
	//[left, keyi - 1] keyi [keyi + 1, right]
	QuickSort1(a, left, keyi - 1);
	QuickSort1(a, keyi + 1, right);
}

3.快速排序挖坑法 

3.1基本思想

        选择最左边的元素作为key,用一个临时变量tmp将key存储起来,用一个left指针指向最左边,用一个right指针指向最右边。right从右往左走找到小于key的值将它放到left处,然后left从左往右走找到大于key的值将它放到right处。重复上述过程,直到left和right相遇,将tmp中保存的key值放到相遇处。然后左右子序列重复上述过程,直到全部元素排好序为止。

        该方法就很自然的在左边挖坑,从右边开始走,不用考虑选择key值从哪边开始走的问题。

3.2挖坑法快速排序的实现

void QuickSort2(int* a, int left, int right)
{
	if (left >= right)
	{
		return;
	}
	int keyi = left;
	int tmp = a[keyi];
	int begin = left;
	int end = right;

	while (begin < end)
	{
		if (begin < end && a[end] >= tmp)
		{
			end--;
		}
		a[begin] = a[end];

		if (begin < end && a[begin] <= tmp)
		{
			begin++;
		}
		a[end] = a[begin];
	}

	a[begin] = tmp;
	keyi = begin;

	QuickSort2(a, left, keyi - 1);
	QuickSort2(a, keyi + 1, right);
}

4. 快速排序前后指针版本

4.1基本思想

4.2快速排序前后指针版本实现 

void QuickSort3(int* a, int left, int right)
{
	if (left >= right)
	{
		return;
	}

	//cur指针找小,找到小于key的,++prev,然后交换
	int keyi = left;
	int prev = left;
	int cur = prev + 1;
	while (cur <= right)
	{
		if (a[cur] < a[keyi] && ++prev != cur)	//判断条件的后半部分是防止自身进行交换
		{
			Swap(&a[prev], &a[cur]);
		}
		cur++;	//交换完cur++,a[cur] >= a[keyi] cur也要++,故写在外面合成一句
	}
	Swap(&a[prev], &a[keyi]);
	keyi = prev;

	QuickSort3(a, left, keyi - 1);
	QuickSort3(a, keyi + 1, right);
}

 5.快速排序非递归版本

5.1基本思想

        为什么要实现快速排序的非递归版本,是因为有时可能递归深度太深,会造成栈溢出的风险,用非递归的方式可以有效的避免该风险。在这里我们用栈的方式来实现快速排序的非递归版本。

        对于递归版本来说,如何能够确定递归子序列的范围,是用子序列的left指针和right指针来确定的,所以对于非递归版本,只要我们能确定子序列的left和right就能确定需要进行递归的子序列范围。所以我们用栈来存储子序列的left和right指针。

        首先,将原数组中left和right存入栈中,然后取出,再依次将右子序列的right,left和左子序列的right和left存入栈中,重复上述操作,对每个子序列进行排序,当栈为空时,结束排序,则此时的数组已被排好序了。

        该方法中使用到栈的接口可以参考C语言实现栈。

5.2快速排序非递归版本实现

        因为非递归版本内部实现的逻辑和递归版本是一样的,不同的只是用栈来存储区间,所以在这我将内部的逻辑写成一个子函数的形式。

        该代码是hoare版本的子函数形式,返回值为上一次单趟之后的keyi。

int PartSort1(int* a, int left, int right)
{
    int keyi = left;
	int begin = left;
	int end = right;
	while (begin < end)
	{
		//右边找小
		while (begin < end && a[end] >= a[keyi])	//加等于方式找到与key相同的数进入死循环
		{
			end--;
		}

		//左边找大
		while (begin < end && a[begin] <= a[keyi])
		{
			begin++;
		}

		Swap(&a[begin], &a[end]);
	}
	Swap(&a[end], &a[keyi]);

	return begin;
}
void QuickSortNonR(int* a, int left, int right)
{
	ST st;
	STInit(&st);
	STPush(&st, right);
	STPush(&st, left);

	while (!(STEmpty(&st)))
	{
		int begin = STTop(&st);
		STPop(&st);
		int end = STTop(&st);
		STPop(&st);

		int keyi = PartSort1(a, begin, end);
		if (keyi + 1 < end) //区间里的值大于1个
		{
			STPush(&st, end);
			STPush(&st, keyi + 1);
		}

		if (begin < keyi - 1)
		{
			STPush(&st, keyi - 1);
			STPush(&st, begin);
		}

	}

	STDestroy(&st);
}

6.快速排序的优化

6.1三数取中

        当key值每次都为数组中的最大值或者最小值的话,则要递归n次,快速排数的时间复杂度就会退化为O(N^2)。为了避免取到的key为最大值或者最小值,对取key进行了一个优化,取数组中left,right以及中间mid指向的元素中,三个元素大小为中间的那个元素,与left交换。

6.1.2三数取中代码实现

//三数取中
int GetMidi(int* a, int left, int right)
{
	int midi = (left + right) / 2;
	if (a[midi] > a[left])
	{
		if (a[midi] < a[right])
		{
			return midi;
		}
		else
		{
			if (a[left] > a[right])
			{
				return left;
			}
			else
			{
				return right;
			}
		}
	}
	else
	{
		if (a[midi] > a[right])
		{
			return midi;
		}
		else
		{
			if (a[right] < a[left])
			{
				return right;
			}
			else
			{
				return left;
			}
		}
	}
}

6.2小区间优化

         对于快速排序的递归来说,当在递归到最后一层时,递归的次数是整个递归的一般,递归创建栈帧和销毁栈帧也会有一定的消耗,并且在元素少的时候,时间复杂度为O(NlogN)的快速排序算法和时间复杂度为O(N^2)的插入排序的消耗差不多,所以在小区间的时候,我们选择用插入排序来代替快速排序,以此来减少递归栈帧的创建和销毁,提高程序性能。

	if ((right - left + 1) < 20)
	{
		InsertSort(a + left, right - left + 1);
	}

        上述代码表示,区间元素的个数小于20个时,采用插入排序进行排序。

6.3优化后的代码

        首先,我先把上述介绍的三种快速排序的版本写成子程序的版本:

// 快速排序hoare版本
int PartSort1(int* a, int left, int right)
{
	//三数取中
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int keyi = left;
	int begin = left;
	int end = right;
	while (begin < end)
	{
		//右边找小
		while (begin < end && a[end] >= a[keyi])	//加等于方式找到与key相同的数进入死循环
		{
			end--;
		}

		//左边找大
		while (begin < end && a[begin] <= a[keyi])
		{
			begin++;
		}

		Swap(&a[begin], &a[end]);
	}
	Swap(&a[end], &a[keyi]);

	return begin;
}
// 快速排序挖坑法
int PartSort2(int* a, int left, int right)
{
	//三数取中
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int keyi = left;
	int tmp = a[keyi];
	int begin = left;
	int end = right;

	while (begin < end)
	{
		if (begin < end && a[end] >= tmp)
		{
			end--;
		}
		a[begin] = a[end];

		if (begin < end && a[begin] <= tmp)
		{
			begin++;
		}
		a[end] = a[begin];
	}

	a[begin] = tmp;

	return begin;
}
// 快速排序前后指针法
int PartSort3(int* a, int left, int right)
{
	//三数取中
	int midi = GetMidi(a, left, right);
	Swap(&a[left], &a[midi]);

	//cur指针找小,找到小于key的,++prev,然后交换
	int keyi = left;
	int prev = left;
	int cur = prev + 1;
	while (cur <= right)
	{
		if (a[cur] < a[keyi] && ++prev != cur) //判断条件的后半部分是防止自身进行交换
		{
			Swap(&a[prev], &a[cur]);
		}
		++cur; //交换完cur++,a[cur] >= a[keyi] cur也要++,故写在外面合成一句
	}
	Swap(&a[prev], &a[keyi]);
	return prev;
}

        然后写一个快速排序的程序来调用上述三个版本并加上优化:

void QuickSort(int* a, int left, int right)
{
	if (left >= right) //一个值的区间和不存在的区间都是结束条件
	{
		return;
	}

	//小区间优化
	if ((right - left + 1) < 20)
	{
		InsertSort(a + left, right - left + 1);
	}
	else
	{
		int keyi = PartSort1(a, left, right);
		QuickSort(a, left, keyi - 1);
		QuickSort(a, keyi + 1, right);
	}
}

        如果要修改调用的子程序,将PartSort1修改为PartSort2或者PartSort3即可。

优化后的性能比较:

7.快速排序的特性总结

(1)  快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫 快速 排序
(2)  时间复杂度: O(N*logN).
(3). 空间复杂度: O(logN).
(4). 稳定性:不稳定.

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

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

相关文章

Linux下如何使用Curl进行网络请求

在Linux系统上&#xff0c;Curl是一个非常强大的网络请求工具&#xff0c;可以用于发送各种类型的HTTP请求&#xff0c;并获取响应结果。它支持常见的HTTP方法&#xff0c;如GET、POST、PUT、DELETE等&#xff0c;还支持HTTPS、FTP等不同协议。Curl提供了丰富的参数选项&#x…

多智能体新进展||斯坦福大学提出新模型‘Hypothetical Minds‘,让AI更懂人类思维

AI论文解读轻松掌握AI前沿技术进展&#xff0c;实时追踪AI动态&#xff0c;互动交流&#xff0c;共同成长进步 标题&#xff1a;Hypothetical Minds: Scaffolding Theory of Mind for Multi-Agent Tasks with Large Language Models 作者&#xff1a;Logan Cross, Violet Xia…

[数据集][图像分类]超声波肾脏结石分类数据集9416张2类别

数据集类型&#xff1a;图像分类用&#xff0c;不可用于目标检测无标注文件 数据集格式&#xff1a;仅仅包含jpg图片&#xff0c;每个类别文件夹下面存放着对应图片 图片数量(jpg文件个数)&#xff1a;9416 分类类别数&#xff1a;2 类别名称:["normal","stone&…

链表(哈希表,有序表)环形链表确定节点的方式

UnOrderedMap UnSortedMap --> C 哈希表&#xff08;无序组织&#xff09; 哈希表如果只有key 没有 value 是HashSet 哈希表如果有key 有 value 是HashMap 哈希表在使用的过程中所有的增删改查都是常数时间&#xff08;比较大&#xff09; 如果存放的是基础类型&#xf…

【网络】套接字(socket)编程——TCP版

接着上一篇文章&#xff1a;http://t.csdnimg.cn/GZDlI 在上一篇文章中&#xff0c;我们实现的是UDP协议的&#xff0c;今天我们就要来实现一下TCP版本的 接下来接下来实现一批基于 TCP 协议的网络程序&#xff0c;本节只介绍基于IPv4的socket网络编程 基于 TCP 的网络编程开…

Java基础入门15:算法、正则表达式、异常

算法&#xff08;选择排序、冒泡排序、二分查找&#xff09; 选择排序 每轮选择当前位置&#xff0c;开始找出后面的较小值与该位置交换。 选择排序的关键&#xff1a; 确定总共需要选择几轮&#xff1a;数组的长度-1。 控制每轮从以前位置为基准&#xff0c;与后面元素选择…

一招搞定异构联邦学习难题:FedKTL的超高效策略!

【联邦学习】在近年来的深度学习领域中备受关注&#xff0c;它通过在保证数据隐私的前提下&#xff0c;协同多个分散的设备或服务器进行模型训练。联邦学习技术能够在不集中数据的情况下&#xff0c;实现数据共享和模型优化&#xff0c;在医疗、金融和智能设备等领域取得了显著…

Linux|centos7|奇怪的知识|perf命令,系统运行瓶颈分析工具

前言&#xff1a; Linux perf 是 Linux 2.6 后内置于内核源码树中的性能剖析&#xff08;profiling&#xff09;工具,它基于事件采样&#xff0c;以性能事件为基础&#xff0c;针对 CPU 相关性能指标与操作系统相关性能指标进行性能剖析&#xff0c;可用于性能瓶颈查找与热点代…

http/sse/websocket 三大协议演化历史以及 sse协议下 node.js express 服务实现打字机案例 负载均衡下的广播实现机制

背景 自从2022年底chatgpt上线后&#xff0c;sse就进入了大众的视野&#xff0c;之前是谁知道这玩意是什么&#xff1f;但是打字机的效果看起来是真的很不错&#xff0c;一度吸引了很多人的趋之若鹜&#xff0c;当然了这个东西的确挺好用&#xff0c;而且实现很简单&#xff0…

Linux环境本地搭建开发工具箱It-Tools并实现公网环境远程使用

文章目录 前言1. 安装Docker2.本地安装部署it-tools3. it-tools工具箱功能—生成docker-compose文件4. 安装cpolar内网穿透5. 固定it-tools公网地址 前言 本篇文章&#xff0c;我们将以Docker方式将IT-Tools部署至本地Linux系统个人服务器&#xff0c;并且结合cpolar内网穿透工…

【无标题】mysql读写分离架构+MyCAT实现读写分离

1、读写分离的目的 数据库负载均衡&#xff1a; 当数据库请求增多时&#xff0c;单例数据库不能够满足业务 需求。需要进行数据库实例的扩容。多台数据库同时相 应请求。也就是说需要对数据库的请求&#xff0c;进行负载均衡 但是由于数据库服务特殊原因&#xff0c;数据库…

安卓用户专属福利:OfficeSuite中文高级版,让你的工作更轻松!

OfficeSuite – 世界顶级移动办公软件&#xff01;Google Play商店下载最多的办公软件应用&#xff0c;迄今为止&#xff0c;智能手机平台上&#xff0c;功能最强大、兼容性最好的移动Office办公套件。创建&#xff0c;查看和编辑Word&#xff0c;Excel和PowerPoint文档&#x…

mysql主从数据库(5.7版本)与python的交互及mycat

mysql数据库基本操作&#xff1a; [rootm ~]# tar -xf mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz 解压压缩包 [rootm ~]# ls anaconda-ks.cfg mysql-5.7.44-linux-glibc2.12-x86_64 mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz [rootm ~]# cp -r mysql-5.7.44-linu…

Ubuntu 批量杀死进程

ps -ef|grep python|grep server|grep -v grep|cut -c 9-16|xargs kill -9这个命令序列是一个在Linux或类Unix系统中使用的脚本片段&#xff0c;用于批量终止&#xff08;强制杀死&#xff09;所有与特定条件&#xff08;这里是包含"python"和"wanghao"的&…

推荐浏览器爬虫插件:Instant Data Scraper 无需写一行代码

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

云计算29-------mysql主从数据库(5.7版本)与python的交互及mycat

mysql数据库基本操作&#xff1a; [rootm ~]# tar -xf mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz 解压压缩包 [rootm ~]# ls anaconda-ks.cfg mysql-5.7.44-linux-glibc2.12-x86_64 mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz [rootm ~]# cp -r mysql-5.7.44-lin…

如何判断树上一个点是否在直径上

# 旅游规划 ## 题目描述 W市的交通规划出现了重大问题&#xff0c;市政府下定决心在全市各大交通路口安排疏导员来疏导密集的车流。但由于人员不足&#xff0c;W市市长决定只在最需要安排人员的路口安排人员。 具体来说&#xff0c;W市的交通网络十分简单&#xff0c;由n个…

【Android Git】Mac配置支持 Gitlab、Gitee和阿里云效多平台

前言 在开发过程中&#xff0c;会遇到多平台项目管理问题&#xff0c;需要进行配置支持&#xff0c;常用的平台有Gitlab、Gitee、阿里云效等&#xff0c;本篇文章记录下使用同一邮箱配置支持的过程。 说明 首先认识下id_ras,一个用于SSH&#xff08;安全外壳协议&#xff09;…

java判断字符串某字符是否为大写/小写/数字?

Character类提供了很多静态方法&#xff0c;用于处理Unicode字符&#xff0c;如下&#xff1a; 也可以将字符转化成小写字母或大写字母。运用如下&#xff1a; 1是数字返回true a不是大写返回false a是小写返回true a转化成大写字母后为A A转化成小写字母后为a

备战金三银四、金九银十、软件测试面试问答

1、问&#xff1a;你在测试中发现了一个bug&#xff0c;但是开发经理认为这不是一个bug&#xff0c;你应该怎样解决&#xff1f; 首先&#xff0c;将问题提交到缺陷管理库里面进行备案。 然后&#xff0c;要获取判断的依据和标准&#xff1a; 根据需求说明书、产品说明、设计…