快速排序(动图详解)(C语言数据结构)

2024-09-03 20:28

本文主要是介绍快速排序(动图详解)(C语言数据结构),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

快速排序:

        快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,其基本思想为:

        任取待排序元素序列中的某元素作为基准值,按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止。
        快速排序有三个版本,将一一实现。

Hoare版本:

        直接上动图理解:

当前为一遍快排。

再分别以key左右两边进行相同的排序

直到key = right = left。

综上得用递归思想。

代码如下:

void QuickSort(int* a, int begin,int end)
{if (begin >= end)//递归返回条件{return;}int right = end;int left = begin;int key = begin;while (begin < end){//找小while (begin<end && a[end]>=a[key]){end--;}//找大while (begin<end && a[begin]<=a[key]){begin++;}swap(&a[begin], &a[end]);}swap(&a[key], &a[begin]);k = begin;QuickSort(a,left,key-1);//key左边递归QuickSort(a,key+1,right);//key右边递归
}

递归展开图:

 优化代码:

        通过画递归展开图后,发现如果这个k的取值一直等于left,有缺点,递归的次数较多。

如果能改进就能让效率提高。

        改进思路:

        如果这个递归能像二叉树一样,总共只需要N*log(N)次,就只需要每次这个k取值去一个中间值,不需要去最大也不需要取最小。

        这样的改进方案,称之为三数取中。 

像刚刚,如果要递归的话一次需要递归9次,如果每次取中间的值当k,每次只需要递归3次 。

等左边递归结束式左边空间还是可以继续利用的。                 

 代码如下:

int GetMid(int* a, int begin, int end)
{int mid = (begin+end) / 2;if (a[mid] > a[end]){if (a[mid] < a[begin]){return mid;}else if (a[end] > a[begin]){return end;}elsereturn begin;}else{if (a[mid] > a[begin]){return mid;}else if (a[end] < a[begin]){return end;}elsereturn begin;}
}
void QuickSort(int* a, int begin,int end)
{if (begin >= end)//递归返回条件{return;}//三数取中int mid = GetMid(a, begin, end);swap(&a[mid], &a[begin]);int right = end;int left = begin;int key = begin;while (begin < end){//找小while (begin<end && a[end]>=a[key]){end--;}//找大while (begin<end && a[begin]<=a[key]){begin++;}swap(&a[begin], &a[end]);}swap(&a[key], &a[begin]);key = begin;QuickSort(a,left,key-1);//key左边递归QuickSort(a,key+1,right);//key右边递归
}

挖坑法版本:

        

代码如下:

void QuickSort2(int* a, int begin, int end)
{if (begin >= end)//递归返回条件{return;}//三数取中int mid = GetMid(a, begin, end);swap(&a[mid], &a[begin]);int left = begin;int right = end;int key = a[left];int hole = left;//第一个为坑while (begin < end){//找小while (begin < end && a[end] >= key){end--;}//找到小的后将数据移到刚刚坑位,并将当前位置作为坑a[hole] = a[end];hole = end;//找大while (begin < end && a[begin] <= key){begin++;}//找到大的后将数据移到刚刚坑位,并将当前位置作为坑a[hole] = a[begin];hole = begin;}a[hole] = key;QuickSort(a, left, hole - 1);//key左边递归QuickSort(a, hole + 1, right);//key右边递归
}

前后指针法:

cur找小,找到以后prev+1后交换cur和prev。

等cur == n-1之后,将a[key]和a[prev]交换,并且将key = prev。 

这个过程保证了prev和cur之间的值是大于key的。

代码如下:

void QuickSort3(int* a, int begin,int end)
{if (begin>=end)//递归返回条件{return;}//三数取中int mid = GetMid(a, begin, end);swap(&a[mid], &a[begin]);int prev = begin;int cur = prev + 1;int key = prev;while (cur<=end-begin){if (a[cur] < a[key]){prev++;if (prev < cur){swap(&a[prev],&a[cur]);}}cur++;}swap(&a[key], &a[prev]);key = prev;QuickSort(a, begin, key - 1);//key左边递归QuickSort(a, key + 1, end);//key右边递归
}

这篇关于快速排序(动图详解)(C语言数据结构)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1133957

相关文章

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示

Python标准库之数据压缩和存档的应用详解

《Python标准库之数据压缩和存档的应用详解》在数据处理与存储领域,压缩和存档是提升效率的关键技术,Python标准库提供了一套完整的工具链,下面小编就来和大家简单介绍一下吧... 目录一、核心模块架构与设计哲学二、关键模块深度解析1.tarfile:专业级归档工具2.zipfile:跨平台归档首选3.

idea的终端(Terminal)cmd的命令换成linux的命令详解

《idea的终端(Terminal)cmd的命令换成linux的命令详解》本文介绍IDEA配置Git的步骤:安装Git、修改终端设置并重启IDEA,强调顺序,作为个人经验分享,希望提供参考并支持脚本之... 目录一编程、设置前二、前置条件三、android设置四、设置后总结一、php设置前二、前置条件

python中列表应用和扩展性实用详解

《python中列表应用和扩展性实用详解》文章介绍了Python列表的核心特性:有序数据集合,用[]定义,元素类型可不同,支持迭代、循环、切片,可执行增删改查、排序、推导式及嵌套操作,是常用的数据处理... 目录1、列表定义2、格式3、列表是可迭代对象4、列表的常见操作总结1、列表定义是处理一组有序项目的

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

SQL Server 中的 WITH (NOLOCK) 示例详解

《SQLServer中的WITH(NOLOCK)示例详解》SQLServer中的WITH(NOLOCK)是一种表提示,等同于READUNCOMMITTED隔离级别,允许查询在不获取共享锁的情... 目录SQL Server 中的 WITH (NOLOCK) 详解一、WITH (NOLOCK) 的本质二、工作

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原