[ 数据结构 -- 手撕排序算法第六篇 ] 快速排序

news2025/1/10 1:47:00

文章目录

  • 前言
  • 一、常见的排序算法
  • 二、快速排序的基本思想
  • 三、快速排序的不同实现
    • 1.hoare版本
    • 2. 挖坑法
    • 3. 前后指针法
    • 4.三种版本单趟排序结果
    • 5.快速排序三数取中优化
    • 6.小区间优化
  • 四、快速排序的特性总结


前言

手撕排序算法第六篇:快速排序!
从本篇文章开始,我会介绍并分析常见的几种排序,例如像插入排序,冒泡排序,希尔排序,选择排序,快速排序,堆排序,归并排序等等!
这篇文章我先来给大家手撕一下快速排序

大家可以点下面的链接去阅读其他的排序算法:
C语言手撕排序算法


正文开始!

一、常见的排序算法

在这里插入图片描述

二、快速排序的基本思想

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

三、快速排序的不同实现

1.hoare版本

我们先来看看动图简单了解一下!
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
如果右边做key的话,要让左边先走,这样能保证相遇位置比key的值大。

在这里插入图片描述
代码如下

//单趟快速排序
int PartSort(int* a,int left,int right)
{
	int keyi = left;
	while (left<right)
	{
		//加上=的判断是因为left和right的值相同就会导致死循环(如 5 2 1 5)
		while (left < right&&a[keyi] <= a[right])
		{
			right--;
		}
		while (left < right && a[keyi] >= a[left])
		{
			left++;
		}
		Swap(&a[left],&a[right]);
	}
	Swap(&a[left],&a[keyi]);
	return left;
}

void QuickSort(int* a,int begin,int end)
{
	if (begin >= end)
		return;
	int keyi = PartSort(a,begin,end);
	//[begin,keyi-1] keyi [keyi+1,end]
	QuickSort(a, begin, keyi - 1);
	QuickSort(a, keyi + 1, end);
}

在这里插入图片描述

2. 挖坑法

先来看动图演示

在这里插入图片描述

简单实例:
在这里插入图片描述
同样的道理我就不给大家演示了哈。

相比第一个版本,挖坑法更好理解

  1. 不需要理解为什么最终相遇位置比key小
  2. 不需要理解为什么左边做key,右边先走!

本质上没有什么区别!

//挖坑法
int PartSort2(int* a, int left, int right)
{
	int key = a[left];
	int pit = left;
	while (left < right)
	{
		//右边先走找小
		while (left < right &&a[right] >=key)
		{
			--right;
		}
		a[pit] = a[right];
		pit = right;
		while (left < right && a[left] <= key)
		{
			++left;
		}
		a[pit] = a[left];
		pit = left;
	}
	a[left] = key;
	return left;
}
void QuickSort(int* a,int begin,int end)
{
	if (begin >= end)
		return;
	int keyi = PartSort2(a,begin,end);
	//[begin,keyi-1] keyi [keyi+1,end]
	QuickSort(a, begin, keyi - 1);
	QuickSort(a, keyi + 1, end);
}
void TestQuickSort()
{
	int a[] = { 4,2,7,8,5,1,0,6 };
	printf("排序前:");
	PrintArray(a, sizeof(a) / sizeof(a[0]));
	QuickSort(a, 0,sizeof(a) / sizeof(a[0])-1);
	printf("排序后:");
	PrintArray(a, sizeof(a) / sizeof(a[0]));
}

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

在这里插入图片描述

3. 前后指针法

先来看一下动态图简单理解一下

(2条消息) 前后指针法 -- 快速排序-CSDN直播

在这里插入图片描述

int PartSort3(int* a, int left, int right)
{
	int key = a[left];
	int prev = left;
	int cur = prev + 1;
	while (cur<=right)
	{
		if (a[cur] < key&&a[++prev]!=a[cur])
			Swap(&a[prev], &a[cur]);
		cur++;
	}
	Swap(&a[left],&a[prev]);
	return prev;
}
void QuickSort(int* a,int begin,int end)
{
	if (begin >= end)
		return;
	//int keyi = PartSort1(a,begin,end);
	//int keyi = PartSort2(a, begin, end);
	int keyi = PartSort3(a,begin,end);
	//[begin,keyi-1] keyi [keyi+1,end]
	QuickSort(a, begin, keyi - 1);
	QuickSort(a, keyi + 1, end);
}
void TestQuickSort()
{
	int a[] = { 4,2,7,8,5,1,0,6 };
	printf("排序前:");
	PrintArray(a, sizeof(a) / sizeof(a[0]));
	QuickSort(a, 0,sizeof(a) / sizeof(a[0])-1);
	printf("排序后:");
	PrintArray(a, sizeof(a) / sizeof(a[0]));
}

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

在这里插入图片描述

选右边做key值。

在这里插入图片描述

4.三种版本单趟排序结果

在这里插入图片描述

5.快速排序三数取中优化

在这里插入图片描述
如果是有序或者接近有序的话,对于快排的时间复杂度就是O(N^2).这个时候还存在栈溢出的问题。

针对key进行优化即可

  1. 随机选key
  2. 三数取中(key选最左边,最右边,中间,选不是最大的,也不是最小的)

代码如下:

int GetMidIndex(int* a,int left,int right)
{
	int mid = (left + right) >> 1;
	if (a[left] <a[mid])
	{
		if (a[mid] < a[right])
		{
			return mid;
		}
		else if (a[left] > a[right])
		{
			return left;
		}
		else
		{
			return right;
		}
	}
	else  //a[left] >a[mid]
	{
		if (a[mid] > a[right])
		{
			return mid;
		}
		else if (a[right]>a[left])
		{
			return left;
		}
		else
		{
			return right;
		}
	}
}

6.小区间优化

在这里插入图片描述
根据快速排序的递归展开图我们可以知道到最后几层每个区间只有几个数据的时候,递归展开图所消耗的栈帧也很多。
所以对于这种情况,我们对于小区间进行优化,因为经过前面的快速排序,小区间已经有序或者接近有序,这样的话就可以使用直接插入排序进行优化!

减少区间很小时,不再使用递归划分的思想让他有序,而是直接使用插入排序对小区进行排序,减少递归调用。

void QuickSort2(int* a, int begin, int end)
{
	if (begin >= end)
		return;
	//小区间直接插入排序控制有序
	if (end - begin +1<= 10)
	{
		InsertSort(a+begin,end-begin+1);
	}

	int keyi = PartSort3(a, begin, end);
	//[begin,keyi-1] keyi [keyi+1,end]
	QuickSort(a, begin, keyi - 1);
	QuickSort(a, keyi + 1, end);
}

四、快速排序的特性总结

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

(本章完!)

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

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

相关文章

JavaSE面试题(二)

1&#xff1a;说一说八大基本数据类型 2&#xff1a;面向对象 面向对象的核心&#xff0c;就是类和对象。Java中的面向对象的思想&#xff1a;万物皆对象。 类&#xff1a;是对一类事物的描述&#xff0c;是抽象的&#xff0c;看不见&#xff0c;摸不着。 对象&#xff1a;是实…

week 7 吴恩达 调参 ,batch norm,softmax

文章目录前言7.1调整参数的过程 Turing progress7.2、scale7.3 如果在实践中探寻超参数7.4 batch normalization 批量归一化7.5 将BN算法拟合到神经网络中7.6 为什么 BN有效&#xff1f;7.7测试时的BN7.8 7.9 softmax regression7.10深度学习的框架前言 7.1调整参数的过程 Turi…

CentOS 8:环境变量

环境变量 环境变量&#xff0c;就是放在当前环境中的变量 无论Linux &#xff0c;还是Windows&#xff0c;都有环境变量 比如&#xff0c;最常用的环境变量 PATH, JAVA_HOME 定义环境变量 export JAVA_HOME/opt/jdk1.8 显示环境变量 echo $JAVA_HOME 查看所有环境变量…

c语言位操作和变量存储类型

c语言位操作 c语言变量存储类型 格式[存储类型说明符] 数据类型说明符 变量名&#xff0c;例如&#xff0c;auto int a;但一般情况下auto是省略的 其他类型说明符还有&#xff1a;static 、extern、register auto最普通动态存储&#xff0c;但所在范围的函数程序结束后&#xf…

处理模型视图中的选择

有关在视图中选择的项的信息存储在QItemSelectionModel类中&#xff0c;这将维护单个模型中项的模型索引&#xff0c;并且独立于任何视图。由于一个模型可以有多个视图&#xff0c;因此可以在视图之间共享选择&#xff0c;从而允许应用程序以一致的方式显示多个视图 选择由选…

Redis高并发锁(二)乐观锁

文章目录redis乐观锁1. watch 监控key2. multi 开启事务3. exec 执行事务4. 演示1) 先用两个连接AB访问redis2) A监控key,此时库存是45013) A开启事务&#xff0c;并且将库存-1,事务进入队列等待执行4&#xff09;此时B更新库存为20015&#xff09;A开始执行事务业务改造1. Sto…

C++11--包装器与可变参数摸板

文章目录可变参数模板递归函数方式展开参数包函数包装器举个例子bind函数举个例子可变参数模板 C11的新特性可变参数模板能够让我们创建可以接受可变参数的函数模板和类模板 // Args是一个模板参数包&#xff0c;args是一个函数形参参数包 // 声明一个参数包Args... args&…

Spring Cloud之Feign消费和Hystrix熔断

Spring Cloud的Feign消费和Hystrix熔断 现如今&#xff0c;由于互联网的流行&#xff0c;很多特产都可以在网上订购&#xff0c;你可以在堆满积雪的冬北订购海南的椰子&#xff0c;海南的椰子就会采用很快的物流方式调送到堆满积地的东北&#xff0c;就相当于在本地实现了买椰…

Opencv(C++)笔记--图像金字塔

目录 1--图像金字塔的原理 2--图像金字塔的用途 3--Opencv API 3-1--拉普拉斯金字塔上采样 3-2--高斯金字塔下采样 3-3--代码实例 4--参考 1--图像金字塔的原理 图像金字塔常用于图像缩放&#xff08;resize&#xff09;和图像分割当中&#xff0c;不同分辨率的图像以金…

CSS基础总结(四)浮动

文章目录 一、为什么需要浮动 1.传统网页布局的三种方式 2.标准流 二、浮动的概述 三、浮动的特性 1.脱标 2.一行显示&#xff0c;顶部对齐 3.具备行内块元素特性 四、清除浮动 1.为什么要清除浮动 2.清除浮动的本质与策略 3.清除浮动的四大方法 &#xff08;1&…

解决安卓刷新recyclerView时导致itemDecoration分栏标题绘制错乱(重叠和隔空现象)

安卓的 itemDecoration 装饰器是个好东西&#xff0c;可以与adapter适配器一样闪耀。但是刷新的时候有可能发生重叠绘制或者莫名隔空的BUG。 三、原作 本文分栏标题装饰器的原作者为简书博主endeavor等人&#xff1a; https://www.jianshu.com/p/8a51039d9e68 二、隔空 紧…

Java+MySQL基于ssm的残疾人管理系统

我国残疾人人口数量相当巨大,据中残联给出的数据,我国约有8500万残疾人。残疾人是社会弱势群体,并且数量庞大影响人数众多,如何能更好的对这些残疾人进行关注和帮助他们更好的生活是当下社会研究的一个主要问题之一,于是我们提出了残疾人信息管理系统的设计与开发。 本课题是一…

内核驱动修改内存

概述 本文会利用内核驱动进行读写取第三方应用内存。 内核实现会使用内联汇编 所以对于内核数据结构每个windwos版本不一样需要判断&#xff0c;本文使用19041所写代码。 命令行&#xff1a;winver 即可查看你当前的版本&#xff0c;如下图19042.631 就是构建版本号 或者调用…

痞子衡嵌入式:低功耗高性能边缘人工智能应用的新答案 - MCXN947

大家好&#xff0c;我是痞子衡&#xff0c;是正经搞技术的痞子。今天痞子衡给大家介绍的是恩智浦MCX系列MCU的新品MCXN947。 自 2015 年恩智浦和飞思卡尔合并成新恩智浦之后&#xff0c;关于它们各自的 Arm Cortex-M 内核通用微控制器代表作系列 LPC 和 Kinetis 接下来怎么发展…

数据结构 | 链式二叉树【递归的终极奥义】

递归——这就是俄罗斯套娃吗&#x1f62e;&#x1f333;链式二叉树的结构及其声明&#x1f333;链式二叉树的四种遍历方式&#x1f4d5;先序遍历&#xff08;先根遍历&#xff09;递归算法图解&#x1f4d5;中序遍历&#xff08;中根遍历&#xff09;&#x1f4d5;后序遍历&…

TIA PORTAL 导出导入数据块

1.导出&#xff1a;选择要导出的数据块鼠标右键-->从块生成源-->仅所选块或包含所有关联块-->最后选择数据块的存储路径保存 2.导入&#xff1a;选外部源文件-->添加新的外部文件-->选择要导入的数据块文件-->单击文件鼠标右键-->从源生成块&#xff0c;最…

Vue3——ref(),reactive(),watch(),computed()的使用

都需要先引入才能使用 ref()函数 作用&#xff1a;创建一个响应式变量&#xff0c;使得某个变量在发生改变时可以同步发生在页面上 模板语句中使用这个变量时可以直接使用变量名来调用&#xff0c;在setup内部调用时则需要在变量明后面加上一个.value获取它的值&#xff0c;原…

记录一次使用卷积神经网络进行图片二分类的实战

写在前面 笔者目前就读的专业是软件工程&#xff0c;并非人工智能专业&#xff0c;但是由于对人工智能有兴趣&#xff0c;于是课下进行了一些自学。正巧最近有些闲暇时间&#xff0c;就想着使用自学的内容做个小型的实战。这篇文章的主要目的也就是从一个入门者的角度&#xf…

【C++】list

本期就来讲讲list的使用技巧 文章目录list的介绍及使用list的介绍list迭代器失效list的模拟实现list与vector的对比我们前面知道迭代器是一个像指针一样的东西&#xff0c;但是在C里面&#xff0c;出来string和vector&#xff0c;其他类都不能 将迭代器当成指针使用&#xff0c…

二叉树的非递归与相关oj

&#x1f9f8;&#x1f9f8;&#x1f9f8;各位大佬大家好&#xff0c;我是猪皮兄弟&#x1f9f8;&#x1f9f8;&#x1f9f8; 文章目录一、二叉树相关oj①二叉搜索树与双向链表②前序遍历和中序遍历构造二叉树二、二叉树的非递归①前序遍历非递归②中序遍历非递归③后序遍历非…