每日OJ题_算法_分治快排②_力扣912. 排序数组(三路划分快排)

目录

三路划分快排原理

[力扣912. 排序数组](#力扣912. 排序数组)

解析代码


三路划分快排原理

分治就是分而治之,快排在数据结构也学过了,现在来学一学三路划分快排(数组划分三块):

前面我们已经实现了三个版本的快速排序的算法,分别是hoare法,挖坑法和前后指针法。但是前面的三个版本的快速排序在某些极端场景中效率都会变得很低,例如大部分都是同一个数的时候,前面的三种方法都不能很高效地完成排序,时间复杂度退化成了O(N^2),所以有必要对之前的排序方法进行一些改进,使它能够适应包含任何数据的数组。于是就有大佬想出了一个更牛的方法,那就是三路划分快排。

三路划分快排的思路: 三路划分,顾名思义就是把数组分成三个部分进行排序,在待排序数组中随机选定一个key,把数组分成小于key的,等于key的,还有大于key的。

具体操作是:

  • 定义一个left下标为数组首元素下标的前一个位置,即left=begin-1;
  • 再定义一个right下标为数组最后一个元素的下标的后一个位置,即right=end+1;
  • 再定义一个下标cur=left,作为遍历数组的下标,一共就left,cur,right三个下标。
  • 从a[cur]开始与key比较,如果a[cur]<key,就用a[cur]和a[++left]交换(记住这里是先++,后交换),然后++cur;如果a[cur]>key,就用a[cur]和a[- -right]交换(记住这里是先减减,后交换);如果a[cur]==key,那就只++cur即可,
  • 这样循环往复,直到cur=right就结束,最后得到的a[begin,left]是比key小的数,a[left+1,right-1]是等于key的数,a[right,end]是大于key的数,划分成三路之后,中间这一路就不用再进行排序了,因为中间这一路都是等于key的,所以已经有序了,只需要对左路和右路再进行递归排序即可。

力扣912. 排序数组

912. 排序数组

难度 中等

给你一个整数数组 nums,请你将该数组升序排列。

示例 1:

复制代码
输入:nums = [5,2,3,1]
输出:[1,2,3,5]

示例 2:

复制代码
输入:nums = [5,1,1,2,0,0]
输出:[0,0,1,1,2,5]

提示:

  • 1 <= nums.length <= 5 * 10^4
  • -5 * 10^4 <= nums[i] <= 5 * 10^4
cpp 复制代码
class Solution {
public:
    vector<int> sortArray(vector<int>& nums) {

};

解析代码

又是这题,C语言用八大排序测试过了,现在再加三指针的快排:

(随机选key在《算法导论》里被证明了最能让快排接近O(N*logN))速度的选key方式,而且下面的三指针可以让快排最慢的情况:排序重复元素O(N^2),直接优化成O(N):

cpp 复制代码
class Solution {
public:
    vector<int> sortArray(vector<int>& nums) {
        srand(time(nullptr)); // 随机数种子
        qsort(nums, 0, nums.size() - 1);
        return nums;
    }

    void qsort(vector<int>& nums, int l, int r)
    {
        if(l >= r)
            return;

        int key = nums[rand() % (r - l + 1) + l]; // 随机选key
        int i = l, left = l - 1, right = r + 1;
        while(i < right)
        {
            if(nums[i] < key)
                swap(nums[++left], nums[i++]);
            else if(nums[i] == key)
                ++i;
            else // >key
                swap(nums[--right], nums[i]);
        }
        // [l, left] [left + 1, right - 1] [right, r]
        qsort(nums, l, left);
        qsort(nums, right, r);
    }
};
相关推荐
古译汉书几秒前
【IoT死磕系列】Day 3:学习HTTP!实战:STM32手写GET请求获取天气实战(附源码+八股文)
数据结构·stm32·物联网·网络协议·学习·算法·http
喜欢吃燃面6 分钟前
基础算法:枚举(上)
c++·学习·算法
郝学胜-神的一滴13 分钟前
计算思维:数字时代的超级能力
开发语言·数据结构·c++·人工智能·python·算法
兵哥工控14 分钟前
mfc 线程启动、挂起、恢复、停止实例
c++·mfc·线程
m0_5312371714 分钟前
C语言-数组练习
c语言·开发语言·算法
识君啊20 分钟前
Java 动态规划 - 力扣 零钱兑换与完全平方数 深度解析
java·算法·leetcode·动态规划·状态转移
xiaoye-duck22 分钟前
《算法题讲解指南:优选算法-滑动窗口》--09长度最小的子数串,10无重复字符的最长字串
c++·算法
二年级程序员34 分钟前
一篇文章掌握“双向链表”
c语言·数据结构·链表
白太岁34 分钟前
Muduo:(5) 主 Reactor 之 Acceptor 与 SubReactor 的分发
服务器·网络·c++·网络协议·tcp/ip
Frostnova丶36 分钟前
LeetCode 762 二进制表示中质数个计算置位
算法·leetcode