一个程序员一生中可能会邂逅各种各样的算法,但总有那么几种,是作为一个程序员一定会遇见且大概率需要掌握的算法。今天就来聊聊这些十分重要的“必抓!”算法吧~
这是快排中的经典算法题,但是很多人从没有对过,涉及到核心问题没搞清楚,不理解想不明白与快速排序的关系是啥??还有好多是直到原理,但是代码写不出来,这节我们就那他练手
参考题目介绍:215. 数组中的第K个最大元素 - 力扣(LeetCode)
这个问题用堆解决也是不错的选择,这个我们稍后分析,我们先看看这个如何基于快速排序来处理的,这个题目出现的频率也是很高的,甚至在很多时候,面试官会直接要求你基于快速排序来解决问题。而我们要直接改造一下上面的快速排序,而不是另起炉灶,只有这样平时练习才有效果,想想怎么能用快速排序解决问题呢?我哦们还是看看这个排序序列:
我们以关键序列{26,53,48,15,13,48,32,15}看一下一次划分的过程:
上面圈起来的位置便是当前已经被赋值给了pivot或者其他位置,可以空出来放移动来的新元素。我们可以看到26最终被放在了属于它的位置上面,不会再变化了,而26的左右两侧可以分别再次进行排序。
这里还有一个关键的信息,我们直到26的索引值为3,也就是说递增排序之后一定是第4大的元素。这就是问题关键。既然我们直到26是第四大,那么我们找第2大,一定实在右边找。找第6大一定是再左边找(当然,如果降序的话就会翻过来),而不需要的那部分就不用关心了,这就是为什么采用快速排序可以解决。
/**
* 快排倒叙注意
* @param nums
* @param left
* @param right
*/
public static void quicksort(int[] nums, int left, int right) {
int start = left;
int end = right;
int pivot = nums[(start + end) >> 1];
// 采用类似前序 [注意倒叙相反]
while(start < end){
// 大于放左边
while (nums[start] > pivot){
start++;
}
// 小于放右边
while (nums[end] < pivot){
end--;
}
// 提前退出循环 即pivot左边大于等于哨兵值,右边小于等于哨兵值
if (start >= end){
break;
}
//交换
int temp = nums[start];
nums[start] = nums[end];
nums[end] = temp;
// 如果左边值和哨兵相同 则右边移动一位
if (nums[start] == pivot){
end--;
}
if (nums[end] == pivot){
start++;
}
}
// 防止栈溢出
if (start == end){
start++;
end--;
}
// 向左或者向右递归
if (left < end){
quicksort(nums,left,end);
}
if (right > start){
quicksort(nums,start,right);
}
}
提示:快速排序;前序遍历;倒叙处理;手撕快排;快排实战
博主创作不易,如果感觉文章内容对你有所帮助的话不妨三连一下再走呗。你们的支持就是我更新的动力!!!
送上我从秋叶大佬哪里拿来的图表示感谢!