力扣--分治(快速排序)算法题I:颜色分类,排序数组

上篇文章:LeetCode模拟算法精解II:外观数列与数青蛙

目录

1.颜色分类

理解题意

算法原理

2.排序数组

理解题意

算法原理


1.颜色分类

https://leetcode.cn/problems/sort-colors/description/

理解题意

对整数0,1,2进行原地排序,不能使用库内置sort函数,使数组升序。

算法原理

由于需要对三个对象同时排序,所以我们使用分治的思想,如下:

在[0, left]中:全为0

在[left+1, i - 1]中:全为1

在[i, right - 1]中:全为等待扫描的元素

在[right, n - 1]中:全为2

nums[i]为0时,交换left和i中的元素,并且left++,i++,不过需要区分先后。

nums[i]为1时,i++即可。

nums[i]为2时,交换right和i中的元素,right--,但是i不能动,因为i存储被交换的元素,暂时还不知道其元素的内容是什么。

复制代码
class Solution {
public:
    void sortColors(vector<int>& nums) {
        int n = nums.size();   

        for(int left = -1, i = 0, right = n; i < right;)
        {
            if(nums[i] == 0) swap(nums[++left], nums[i++]);
            else if(nums[i] == 1) i++;
            else swap(nums[--right], nums[i]);
        }
    }
};

2.排序数组

https://leetcode.cn/problems/sort-an-array/description/

理解题意

在时间复杂度为o(nlog(N))的前提下,不使用内置函数,将数组变为升序排列。

算法原理

在上一题"颜色分类"中,运用了分类和快速排序的方式,在本题中,则需要在此基础上进行优化--由于时间复杂度的要求。

注意:如果想要时间复杂度接近nlog(n),那么基准元素的取值就应该是取随机值,此结论在算法导论中有所提及。

我们先随机选取一个基准元素key,在此基础上,对数组中的元素进行分类:

此后,如果nums[i] < key:swap(nums[++left], nums[i++])

如果nums[i] = key:i++

如果nums[i] > key:swap(nums[--right], nums[i])

而有关基准元素的选择,我们使用:r = rand();

接着,取其下标:nums[r % (left - right + 1) + left],其中,r % (left - right + 1)表示为[0, n - 1]

代码如下:

复制代码
class Solution 
{
public:
    vector<int> sortArray(vector<int>& nums) 
    {
        srand(time(NULL));
        qsort(nums, 0, nums.size() - 1);
        return nums;
    }

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

        // 数组分三块
        int key = getRandom(nums, l, r);
        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 swap(nums[--right], nums[i]); 
        }

        // [l, left] [left + 1, right - 1], [right, r]
        qsort(nums, l, left);
        qsort(nums, right, r);
    }

    int getRandom(vector<int>& nums, int left, int right)
    {
        int r = rand();
        return nums[r % (right - left + 1) + left];
    }
};

本章完。

相关推荐
2301_822703204 小时前
Flutter 框架跨平台鸿蒙开发 - 创意声音合成器应用
算法·flutter·华为·harmonyos·鸿蒙
cmpxr_4 小时前
【C】数组名、函数名的特殊
c语言·算法
KAU的云实验台4 小时前
【算法精解】AIR期刊算法IAGWO:引入速度概念与逆多元二次权重,可应对高维/工程问题(附Matlab源码)
开发语言·算法·matlab
会编程的土豆5 小时前
【数据结构与算法】再次全面了解LCS底层
开发语言·数据结构·c++·算法
低频电磁之道5 小时前
解决 Windows C++ DLL 导出类不可见的编译错误
c++·windows
大熊背5 小时前
如何利用Lv值实现三级降帧
算法·自动曝光·lv·isppipeline
大尚来也6 小时前
驾驭并发:.NET多线程编程的挑战与破局之道
java·前端·算法
向阳而生,一路生花6 小时前
深入浅出 JDK7 HashMap 源码分析
算法·哈希算法
君义_noip6 小时前
信息学奥赛一本通 4150:【GESP2509七级】⾦币收集 | 洛谷 P14078 [GESP202509 七级] 金币收集
c++·算法·gesp·信息学奥赛·csp-s