看似青铜实则王者很多人提起快排和二分都觉得很容易的样子 , 但是让现场Code很多就翻车了 , 就算可以写出个递归版本的代码 , 但是对其中的复杂度分析、边界条件的考虑、非递归改造、代码优化等就无从下手 , 填鸭背诵基本上分分钟就被面试官摆平了 。

文章插图
那年初识快速排序
快速排序Quicksort又称划分交换排序partition-exchange sort , 简称快排 , 一种排序算法 。最早由东尼·霍尔(C. A. R. Hoare)教授在1960年左右提出 , 在平均状况下 , 排序n个项目要O(nlogn)次比较 。
在最坏状况下则需要O(n^2)次比较 , 但这种状况并不常见 。事实上 , 快速排序通常明显比其他算法更快 , 因为它的内部循环可以在大部分的架构上很有效率地达成 。
快速排序的核心思想在计算机科学中 , 分治法(Divide&Conquer)是建基于多项分支递归的一种很重要的算法范式 , 快速排序是分治思想在排序问题上的典型应用 。
所谓分治思想D&C就是把一个较大规模的问题拆分为若干小规模且相似的问题 。再对小规模问题进行求解 , 最终合并所有小问题的解 , 从而形成原来大规模问题的解 。
字面上的解释是"分而治之" , 这个技巧是很多高效算法的基础 , 如排序算法(归并排序、快速排序)、傅立叶变换(快速傅立叶变换) 。
分治法中最重要的部分是循环递归的过程 , 每一层递归有三个具体步骤:
- 分解:将原问题分解为若干个规模较小 , 相对独立 , 与原问题形式相同的子问题 。
- 解决:若子问题规模较小且易于解决时 , 则直接解 。否则 , 递归地解决各子问题 。
- 合并:将各子问题的解合并为原问题的解 。
快速排序的基本过程
快速排序使用分治法来把一个序列分为小于基准值和大于基准值的两个子序列 。
递归地排序两个子序列 , 直至最小的子序列长度为0或者1 , 整个递归过程结束 , 详细步骤为:
- 挑选基准值: 从数列中挑出一个元素称为基准pivot , 选取基准值有数种具体方法 , 此选取方法对排序的时间性能有决定性影响 。
- 基准值分割: 重新排序数列 , 所有比基准值小的元素摆放在基准前面,所有比基准值大的元素摆在基准后面 , 与基准值相等的数可以到任何一边,在这个分割结束之后 , 对基准值的排序就已经完成 。
- 递归子序列: 递归地将小于基准值元素的子序列和大于基准值元素的子序列排序 , 步骤同上两步骤,递归终止条件是序列大小是0或1 , 因为此时该数列显然已经有序 。
快速排序的递归实现
#include<stdio.h>int a[9]={5,1,9,6,7,11,3,8,4};void exchange(int *p,int *q){ int temp=*p; *p=*q; *q=temp;}int quicksort(int left,int right){ if(left>=right){ return 0; } int i,j,temp; temp=a[left]; i=left; j=right; while(i!=j){ while(i<j&&a[j]>=temp){ j--; } exchange(&a[i],&a[j]);while(i<j&&a[i]<=temp){ i++;} exchange(&a[i],&a[j]); } quicksort(i+1,right); quicksort(left,i-1); }int main(){ quicksort(0,8); for(int i=0;i<=8;i++){ printf("%d ",a[i]); }}
#include<IOStream>using namespace std;template <typename T>void quick_sort_recursive(T arr[], int start, int end) { if (start >= end) return; T mid = arr[end]; int left = start, right = end - 1; //整个范围内搜寻比枢纽值小或大的元素 , 然后左侧元素与右侧元素交换 while (left < right) { //试图在左侧找到一个比枢纽元更大的元素 while (arr[left] < mid && left < right) left++; //试图在右侧找到一个比枢纽元更小的元素 while (arr[right] >= mid && left < right) right--; //交换元素 std::swap(arr[left], arr[right]); } //这一步很关键 if (arr[left] >= arr[end]) std::swap(arr[left], arr[end]); else left++; quick_sort_recursive(arr, start, left - 1); quick_sort_recursive(arr, left + 1, end);}//模板化template <typename T> void quick_sort(T arr[], int len) { quick_sort_recursive(arr, 0, len - 1);}int main(){ int a[9]={5,1,9,6,7,11,3,8,4}; int len = sizeof(a)/sizeof(int); quick_sort(a,len-1); for(int i=0;i<len-1;i++) cout<<a[i]<<endl;}两个版本均可正确运行 , 但代码有一点差异:
推荐阅读
-
挪威@病毒源头“水落石出”!挪威无法继续狡辩,世卫组织正式向全球发声
-
娱乐圈的秘密|武侠仙侠游戏迷速速集合!,酷狗CMJ国风音乐盛宴即将开启
-
-
后劲|原创华谊兄弟连亏两年凭借《八佰》摆脱退市噩梦?偿债压力大后劲乏力 | 中报季
-
-
趣头条|e-tron概念车中国首发,量产版2022年就来了,奥迪Q4
-
梦泪|8号碎片商店皮肤轮换,心灵骇客不加入没事,梦泪:换他秒赚1430
-
-
世代|周世代第一实在人,开始了一个王朝的布局|史记周次元006
-
-
宾利|300万宾利撞上石墩,车主蹲在路旁一脸憋屈,网友:土豪也会心疼钱?
-
在淘宝开店以后什么情况下会被查封账户 如何在淘宝开店
-
-
谈风笑谈|拆二代将全面消失,一类人将会有麻烦,国家正式表态:明年起
-
-
鸽子潮女时尚空间@11岁薇薇安基础穿搭甜美活力,安吉丽娜朱莉墨镜配长裙出街显气场
-
20岁女生怎样跟30多岁未婚同事同性聊天?工作需频繁见面,所以有点尴尬。我性格慢热幼稚
-
-
『老司机不翻车』携手吴宣仪,网友:男神!,肖战人气回升了?《斗罗大陆》将播
-
古树红茶的贮藏方法,速溶茶生产过程中人工调香的方法