算法学习:快速排序

news2024/9/22 3:46:34

在这里插入图片描述

🔥 个人主页:空白诗

在这里插入图片描述

文章目录

    • 🚀 引言
    • 📌 快速排序算法核心思想
      • 1. 选择基准值(Pivot)
      • 2. 分区操作(Partitioning)
      • 3. 递归排序子序列
    • 📌 JavaScript 实现
      • 1. 快速排序主函数
      • 2. 分区函数
      • 3. 示例代码
    • 📌 优化建议
      • 1. 三数取中法
      • 2. 小数组时切换排序算法
      • 3. 尾递归优化
      • 4. 并行化处理
    • 📌 时间复杂度分析
    • 📌 总结

🚀 引言

快速排序(Quick Sort)是一种高效的排序算法,由计算机科学界的传奇人物托尼·霍尔(Tony Hoare)于1960年巧妙地提出。这一算法的核心智慧在于运用了经典的分治法策略——犹如古代兵法中的“分而治之”,将一个错综复杂的大列表分割成两个相对简单的子列表,随后对这两个子列表施以同样的策略,直到每个子列表都只剩下单一元素或为空,此时整个序列自然归于有序。此过程宛如构建一座秩序井然的金字塔,自下而上,层层推进。


📌 快速排序算法核心思想

在这里插入图片描述

1. 选择基准值(Pivot)

这是算法流程的起点,从数列中精心挑选出一个元素,赋予它一个特殊角色——“基准”(pivot)。基准的选择可以很灵活,但理想情况下应倾向于选择一个能将数据集大致均匀分割的值,以促进算法效率。

2. 分区操作(Partitioning)

分区操作是快速排序的精髓所在。其目标是在遍历数列一次的过程中,通过交换元素位置,使得所有小于基准值的元素都位于基准之前,而所有大于基准值的元素都排列在其后相等的元素可以放置在任一侧,完成这一操作后,基准值恰好位于其最终排序后的位置,即序列的中间。这一巧妙划分不仅为后续递归奠定了基础,也直接体现了快速排序分而治之的哲学。

3. 递归排序子序列

基于分区结果,数列被明确划分为两个独立的部分:左侧全部小于基准值,右侧则大于基准值。接下来,算法会对这两个子序列递归地应用同样的排序逻辑。通过不断地将问题规模减半,直到每个子序列只剩下一个或零个元素(这时自然视为已排序),整个数列便会在这一系列递归调用中逐步构建出全局的有序状态。这一策略确保了快速排序高效利用了分治策略的优势,尤其是在平均情况下展现出极高的效率。


📌 JavaScript 实现

1. 快速排序主函数

function quickSort(arr, left = 0, right = arr.length - 1) {
  // 如果左指针小于右指针,说明还有未排序的区间
  if (left < right) {
    // 调用分区函数,返回pivot的索引,完成一次分区
    const pivotIndex = partition(arr, left, right);
    
    // 对pivot左边的子数组进行快速排序
    quickSort(arr, left, pivotIndex - 1);
    // 对pivot右边的子数组进行快速排序
    quickSort(arr, pivotIndex + 1, right);
  }
  
  // 返回排序后的数组,实际上由于是原地排序,此处return并非必要
  return arr;
}

2. 分区函数

function partition(arr, left, right) {
  // 选择最右侧元素作为基准值pivot
  const pivot = arr[right];
  let i = left; // i为小于pivot的元素的边界,初始时指向最左侧
  
  // 遍历除了pivot外的所有元素
  for (let j = left; j < right; j++) {
    // 如果当前元素小于或等于pivot
    if (arr[j] <= pivot) {
      // 交换arr[i]和arr[j],并将i右移一位,保持i左侧的元素都小于等于pivot
      [arr[i], arr[j]] = [arr[j], arr[i]]; // ES6解构赋值进行交换
      i++; 
    }
  }
  
  // 最后将pivot(arr[right])与arr[i]交换,使得pivot位于正确的位置上
  [arr[i], arr[right]] = [arr[right], arr[i]];
  
  // 返回pivot的最终位置索引
  return i;
}

3. 示例代码

// 未排序数组示例
const unsortedArray = [3, 6, 8, 10, 1, 2, 1];

// 打印原始数组
console.log("Unsorted:", unsortedArray);

// 调用快速排序函数,得到排序后的数组
const sortedArray = quickSort(unsortedArray);

// 打印排序后的数组
console.log("Sorted:", sortedArray);

这段代码实现了快速排序算法,通过quickSort函数递归地将数组分为更小的子数组,并通过partition函数完成每部分的排序,最终达到整个数组有序的目的。代码中使用了ES6的解构赋值来简化元素交换的操作,使得代码更加简洁易读。


📌 优化建议

1. 三数取中法

  • 核心思想:通过更智能地选择基准值(pivot)来优化快速排序的性能。
  • 操作步骤
    1. 比较数组首部、中部、尾部的元素。
    2. 选取这三个数中的中位数作为分区操作的基准值。
    3. 这样做能有效平衡划分,即使在数据部分有序的情况下也能保持较好的性能,减少最坏情况的发生概率。

2. 小数组时切换排序算法

  • 适用条件:当待排序序列的元素数量较少时(例如少于10或15个)。
  • 策略详情:快速排序在小数组上的优势不明显,此时切换到插入排序等简单排序算法更为高效。因为插入排序在小数据集上具有较低的常数因子和无需递归的优点,能够快速完成排序,与快速排序形成互补。

3. 尾递归优化

  • 概念阐述:确保递归调用是函数的最后一个操作,便于某些支持该特性的编译器进行优化。
  • 实施方法:设计递归逻辑时,直接在递归调用的返回语句中返回计算结果,避免在递归返回后还需执行其他操作。
  • 注意事项:虽然JavaScript等语言不一定能自动优化尾递归,遵循此原则编写代码依然有助于提高可读性和未来跨平台移植的兼容性。

4. 并行化处理

  • 目标:利用多核CPU资源加速排序过程。
  • 实施步骤
    1. 将大数组分割成多个小块。
    2. 各个CPU核心独立并行地对分块数据执行快速排序。
    3. 最后合并各块已排序的结果。
  • 优势:在拥有多个处理器核心的系统上,此策略能显著缩短排序时间,尤其适合处理海量数据。

通过上述一系列优化措施,快速排序算法不仅在理论上保持了较高的时间效率,在实际应用中也变得更加灵活和健壮,能够有效应对各种规模数据集的排序挑战,展现出更高的性能和稳定性。


📌 时间复杂度分析

在这里插入图片描述

在这里插入图片描述

快速排序算法的性能极大程度上取决于基准值(Pivot)的选择策略。

  • 最优情况:若每次分区操作都能均匀地将数据集切分为两部分,每部分包含近似相等数量的元素,则递归树的深度为log₂n。鉴于每一层递归涉及遍历数组,总体操作计数约为n * log₂n。因此,快速排序在最佳情况下的时间复杂度为O(n log n),表现出高度的效率。

  • 最差情况:相反,若每次选取的基准值都导致极不均衡的分区,极端情形下每次仅将数组分为一个元素和剩余所有元素两部分,这将导致递归深度上升至n,伴随每次遍历数组的操作,时间复杂度恶化为O(n²),与冒泡排序相近。

  • 平均情况:在实践中,若假定分区大致均匀,即每次都能将数据集分为两个大小相似的子集,快速排序的平均时间复杂度同样为O(n log n)。这对于多数随机分布数据集而言,是一个非常高效的排序方案。

鉴于最坏情况下的性能瓶颈,实际部署快速排序算法时,往往配合采用基准值优化策略,比如“三数取中法”,来增强其鲁棒性和普遍适用性,确保在多种数据条件下仍能保持高效的排序性能。


📌 总结

快速排序算法通过分治法策略实现高效排序,其核心包括选择基准值、分区操作及递归排序子序列三大步骤。为了进一步提升性能和适应不同场景,可采纳诸如三数取中法优化基准选择、小数组时切换至插入排序、尾递归优化及并行处理等策略。这些优化不仅能够减少最坏情况出现的概率,还充分利用现代计算资源,使快速排序在实践中表现得更为出色,成为处理大量数据排序任务的优选算法之一。

总之,快速排序凭借其高效与灵活性,在众多排序算法中占据重要地位,广泛应用于各种数据排序需求之中。


在这里插入图片描述

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

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

相关文章

OGG几何内核-BRepBuilderAPI_MakeEdge学习

OGG几何内核fork自OCCT 7.7.0&#xff0c; BRepBuilderAPI_MakeEdge是几何内核的一个重要和基础的功能&#xff0c;也十分复杂&#xff0c;因为要支持line、circle、ellipse&#xff0c;parabola&#xff0c;hyperbola&#xff0c;circle&#xff0c;beziercurve&#xff0c;b…

Web前端开发技术-格式化文本 Web页面初步设计

目录 Web页面初步设计 标题字标记 基本语法&#xff1a; 语法说明&#xff1a; 添加空格与特殊符号 基本语法&#xff1a; 语法说明: 特殊字符对应的代码: 代码解释&#xff1a; 格式化文本标记 文本修饰标记 计算机输出标记 字体font标记 基本语法&#xff1a; 属…

《TortoiseSVN》简单使用说明

##################工作记录#################### 常用图标说明 一个新检出的工作副本 修改过的文件 更新过程遇到冲突的文件 你当前对文件进行了锁定&#xff0c;不要忘记不使用后要解锁&#xff0c;否则别人无法使用 当前文件夹下的某些文件或文件夹已经被调度从版本控制…

BUUCTF靶场 [reverse]easyre、reverse1、reverse2

工具&#xff1a; DIE&#xff1a;下载&#xff1a;https://download.csdn.net/download/m0_73981089/89334360 IDA&#xff1a;下载&#xff1a;https://hex-rays.com/ida-free/ 新手小白勇闯逆向区&#xff01;&#xff01;&#xff01; [reverse]easyre 首先查壳&#xf…

一刷后日谈

后日谈 1.前言 ​ 今天是一刷结束的日子&#xff0c;回顾一路走过来的路&#xff0c;还是得对自己说声谢谢的&#xff1b; ​ 我是一个向来都很抗拒编程的人&#xff0c;那繁琐的符号让我觉得很是头疼&#xff0c;本科期间数据结构与算法都重修了三次&#xff0c;最后临近毕…

Redis --学习笔记

Redis简介 一个基于内存的key-value结构数据库。Redis 是互联网技术领域使用最为广泛的存储中间件 特点&#xff1a; 基于内存存储&#xff0c;读写性能高 适合存储热点数据&#xff08;热点商品、资讯、新闻&#xff09; 企业应用广泛 Redis默认端口号为6379 Redis是用…

Shell之(正则表达式)

目录 一、正则表达式的概括 常用选项 基础正则表达式 查看以xxx为开头 查看以xxx为结尾的文件内容 匹配单个或多个字符 匹配中括号里的内容 对子表达式进行多次或者限定次数的匹配 拓展正则表达式的基本用法 扩展元字符 匹配前面的字符或表达式至少1次&#xff0…

分布式微服务之springboot学习[上]

文章目录 SpringBoot基本介绍官方文档springboot是什么?springboot快速入门需求/图解说明完成步骤快速入门小结 Spring SpringMVC SpringBoot的关系梳理关系如何理解 -约定优于配置 依赖管理和自动配置依赖管理什么是依赖管理修改自动仲裁/默认版本号 starter场景启动器starte…

CSDN智能总结助手

github项目地址&#xff1a; https://github.com/anjude/little-demo/tree/master 获取CSDN的user name和user token 打开csdn&#xff0c;打开控制台 - Application - Cookies&#xff0c;找到domain为blog.csdn.net的cookie&#xff0c;复制user_name和user_token的值 把上…

【计网】广播域和冲突域

一、相关概念 1.各层次设备 2.冲突域 2.1定义 冲突域通俗来讲就是在同一个网络中&#xff0c;两台设备同时传输的话会产生冲突。位于OSI的第一层&#xff1a;物理层 例如在集线器场景下&#xff0c;集线器属于物理层设备&#xff0c;它不具备交换机的功能&#xff0c;当收到节…

PIM Anycast RP(PIM实现)

拓扑图 配置 配置全局使能组播路由 配置OSPF 配置PIM-SM 配置anycast-rp&#xff0c;Loopback 1为Anycast-RP地址&#xff0c;Loopback 0为Anycast-RP本地地址 配置IGMP sysname AR1 # multicast routing-enable # interface GigabitEthernet0/0/0ip address 10.1.12.1 2…

删除重复的电子邮箱-力扣

题目 答案 delete p1 from Person as p1 ,Person as p2 where p1.email p2.email AND p1.id>p2.id; 解析 按照题目删除即可&#xff0c;注意删除的是一条&#xff0c;所以直接是person p1 即可 此处不能用distinct &#xff0c;其一般与select相互使用。

遍历列表

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 遍历列表中的所有元素是常用的一种操作&#xff0c;在遍历的过程中可以完成查询、处理等功能。在生活中&#xff0c;如果想要去商场买一件衣服&#…

C++ RPC ORM 高速解析

支持所有常用编程语 https://capnproto.org/GitHub - capnproto/capnproto: Capn Proto serialization/RPC system - core tools and C library https://capnproto.org/capnproto-c-win32-1.0.2.zip 常用命令&#xff1a; capnp help capnp compile -oc myschema.capn…

Qt代码初识

文章目录 Qt代码初识1. Qt Hello World 程序1.1 使⽤ "按钮" 实现1.1.1 纯代码⽅式实现1.1.2 可视化操作实现 1.2 使⽤ "标签" 实现1.2.1 纯代码⽅式实现1.2.2 可视化操作实现 2. 项⽬⽂件解析2.1 .pro ⽂件解析2.2 widget.h ⽂件解析2.3 main.cpp ⽂件解析…

防火墙技术基础篇:解析防火墙的网络隔离机制

防火墙技术基础篇&#xff1a;解析防火墙的网络隔离机制 网络安全在现代社会中扮演着重要的角色&#xff0c;保护网络系统、用户和数据免受未经授权的访问、破坏和窃取。个人、企业和国家都需要加强网络安全意识&#xff0c;采取有效措施保护自身的网络安全。随着网络攻击手段…

VS2022通过C++网络库Boost.asio搭建一个简单TCP异步服务器和客户端

基本介绍 上一篇博客我们介绍了通过Boost.asio搭建一个TCP同步服务器和客户端&#xff0c;这次我们再通过asio搭建一个异步通信的服务器和客户端系统&#xff0c;由于这是一个简单异步服务器&#xff0c;所以我们的异步特指异步服务器而不是异步客户端&#xff0c;同步服务器在…

SQL靶场搭建

概述 简单介绍一下SQL靶场的搭建&#xff0c;以及在搭建过程中遇到的一些问题。使用该软件搭建靶场相对简单&#xff0c;适合新手小白。当然&#xff0c;也可以在自己的虚拟机下进行搭建&#xff0c;相对来说就较为复杂。本章主要讲解使用Phpstudy进行SQL靶场搭建。 这里我推…

SpringBoot 实现 RAS+AES 自动接口解密

一、讲个事故 接口安全老生常谈了 过年之前做了过一款飞机大战的H5小游戏&#xff0c;里面无限模式-需要保存用户的积分&#xff0c;因为使用的Body传参&#xff0c;参数是可见的。 为了接口安全我&#xff0c;我和前端约定了传递参数是&#xff1a;用户无限模式的积分“我们…

一个简约高级视差效果PR动态图文开场视频模板

这是一个高质量且易于定制的pr模板。具有模块化结构&#xff0c;可以轻松更改内容。包括视频教程&#xff0c;即使是新手小白也可以轻松套用模板制作视频。 主要特点&#xff1a; 水平&#xff08;19201080&#xff09;和垂直&#xff08;10801920&#xff09;分辨率&#xff…