力扣--分治(快速排序)算法题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

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

numsi为1时,i++即可。

numsi为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,在此基础上,对数组中的元素进行分类:

此后,如果numsi < key:swap(nums++left, numsi++)

如果numsi = key:i++

如果numsi > key:swap(nums--right, numsi)

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

接着,取其下标:numsr % (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];
    }
};

本章完。

相关推荐
Tian_Hang3 小时前
C++原型模式(Protype)
开发语言·c++·算法
bIo7lyA8v3 小时前
算法复杂度的渐进分析与实际运行时间的差异的技术8
算法
一切皆是因缘际会3 小时前
LLM轻量化联邦微调机理
数据结构·人工智能·数学建模·ai
yuan199974 小时前
欧拉梁静力与屈曲计算的 MATLAB 实现(有限差分法 + 解析解)
开发语言·算法·matlab
FL16238631294 小时前
[cmake]基于C++使用纯opencv部署ppocrv5v6的onnx模型
开发语言·c++·opencv
玖玥拾4 小时前
C/C++ 数据结构(六)链表迭代器与底层
c语言·数据结构·c++·链表·stl库
牛油果子哥q4 小时前
AVL平衡树与红黑树深度精讲对比,平衡因子、四大旋转原理、着色规则、平衡策略、性能差异与面试手撕全解
数据结构·c++·面试
汉克老师5 小时前
GESP7级C++考试语法知识(二、指数函数(3、综合练习)
c++·算法·数学建模·指数函数·gesp7级·复利
C++ 老炮儿的技术栈5 小时前
Ubuntu root账号自动登陆
linux·运维·服务器·c语言·c++·ubuntu·visual studio
林间码客5 小时前
04 ROC曲线与AUC:从零开始手动计算
大数据·人工智能·算法