当前位置: 首页 > news >正文

C++ 快速排序(Quick Sort)深度精讲:分治思想、Lomuto 分区法及三数取中优化,面试手撕必会

一、算法核心原理(“是什么”与“为什么快”)

快速排序之所以“快”,在于它的分区操作能在一次遍历中将一个元素(基准)放到最终位置上,同时让左右两侧满足大小关系。

相比于希尔排序的“逐渐逼近”,快排采用的是递归分割策略——每次都将问题规模减半,因此平均深度仅为 log₂n。

二、经典 C++ 实现:Lomuto 分区法(最易理解的写法)

这是面试中最推荐手撕的版本,逻辑直观,不易出错。核心是维护一个i指针,指向“小于等于基准的区域的末尾”。

#include <iostream> #include <vector> using namespace std; // 分区函数:将 arr[low..high] 分区,返回基准元素的最终下标 int partition(vector<int>& arr, int low, int high) { int pivot = arr[high]; // 选最右边的元素作为基准 int i = low - 1; // i 指向小于等于 pivot 的区域的最后一个位置 for (int j = low; j < high; j++) { // 当前元素小于等于基准时,将其与 i+1 位置交换,扩大小于区域 if (arr[j] <= pivot) { i++; swap(arr[i], arr[j]); } } // 最后将基准放到中间位置(i+1) swap(arr[i + 1], arr[high]); return i + 1; // 返回基准的最终位置 } // 快速排序递归主函数 void quickSort(vector<int>& arr, int low, int high) { if (low < high) { int pi = partition(arr, low, high); // pi 是基准的索引 quickSort(arr, low, pi - 1); // 递归排左边 quickSort(arr, pi + 1, high); // 递归排右边 } }

三、核心优劣分析(面试必答)

  • 优势 1:极佳的缓存局部性。分区操作是在原数组上顺序遍历,对 CPU 缓存极其友好,常数因子远小于归并排序和堆排序。

  • 优势 2:原地排序(In-place)。除了递归栈开销,不需要额外的大块内存(不像归并需要 O(n) 辅助空间)。

  • 致命短板(最坏情况):当数组已经有序(正序或逆序)且每次选最后一个元素作基准时,分区极度不平衡,递归深度变为 n,时间复杂度退化为O(n²)

四、工程级优化策略(加分项,展示深度)

为了防止最坏情况,C++ 的std::sort(内省排序)混用了快排并加入了保险机制。面试手撕时,你可以主动提出以下几种优化:

  1. 三数取中(Median-of-Three):不选固定端,而是取lowmidhigh三个位置的中位数作为基准,极大降低遇到有序数组时退化的概率。
  2. int mid = low + (high - low) / 2; if (arr[mid] < arr[low]) swap(arr[mid], arr[low]); if (arr[high] < arr[low]) swap(arr[high], arr[low]); if (arr[high] < arr[mid]) swap(arr[high], arr[mid]); swap(arr[mid], arr[high]); // 将中位数放到 high 位置作为 pivot
  3. 小区间插入排序:当子数组长度小于阈值(如 10~15)时,递归开销过大,直接改用插入排序。
  4. 随机化基准int pivot = arr[low + rand() % (high - low + 1)];从概率上规避最坏情况。

五、总结速查

  • 手撕必用:Lomuto 分区法(代码短)。

  • 面试亮点:主动说出“快排是不稳定的”、“有序数组会让它变慢,需要三数取中优化”。

  • 工程警告:写业务代码永远不要手写快排,请直接调用std::sort,因为它是内省排序,已经内置了上述所有优化。

资源推荐

《C++八股文》2026版

贪心算法

http://www.jsqmd.com/news/1128746/

相关文章:

  • STM32L433RC与DC-DC降压转换器设计实战
  • TVA与具身智能的结构性关联(10)
  • 数据产业服务分类(32)——数据产业——数字技术服务与数据产业服务
  • Kafka 消息重试设计:别让失败消息原地打转
  • Modbus工控安全渗透测试:Smod框架实战与防御指南
  • 【camera 005】 Camera Surface 数据流获取流程深度解析
  • 4-20mA电流环技术与XTR116工业应用指南
  • 企业知识库同步延迟:文档更新后,答案不能还停在昨天
  • 数学基础速查——大模型工程师的“最小够用集“
  • 数据产业服务分类(33)——数据产业——政府管理部门
  • Si4732与PIC18F97J94数字广播接收方案设计与优化
  • 5分钟掌握Axure RP中文界面:完整汉化包安装与配置指南
  • 从零开始学AI:小白程序员必备收藏指南,快速掌握大模型实战技能
  • 新手误区:只会调包不懂底层,永远成不了高级AI工程师
  • 终极便携式Windows C/C++开发工具链:w64devkit完全指南
  • ProperTree终极指南:跨平台plist编辑器让配置文件编辑变得简单
  • 找了个开源的 AI 写小说 Agent,自己部署跑了一遍
  • python___let`s try it 3---计算水仙花数
  • ALVR无线串流技术深度解析:实现PC VR游戏无线化自由体验
  • 大模型入门指南:小白程序员必收藏,轻松掌握AI核心技术!
  • 【LE Audio】CSIP精讲[4]:Set Coordinator全流程管控与实现精要
  • Windows风扇控制终极指南:用FanControl打造静音高效的散热系统
  • 2026最新AI Agent从零落地实战指南!小白程序员专属企业级开发教程
  • C++语言基础4:例程讲解(结合在QT的应用)
  • 3步解锁网易云音乐:ncmdump工具让NCM格式不再困扰你
  • Kimi LeetCode 3463. 判断操作后字符串中的数字是否相等 II C++实现
  • 基于Si4731与PIC18F25K50的FM收音系统设计与实现
  • Mi-Create终极指南:免费可视化小米手表表盘制作工具完整教程
  • REPENTOGON实战深度配置指南:解锁以撒结合终极扩展能力
  • 技术革命:EmojiOne Color如何重塑表情符号的跨平台标准