排序数组 ---- 分治-快排

题目链接

题目:

分析:

  • 回顾一下快速排序, 快速排序的思想就是找一个基准值key, 按照基准值,将数组分成两块, 分别是<key和>key的区域,然后继续对这两个区域划分, 那么快速排序的时间复杂度是O(N*logN), 但是如果数组中有许多相同的元素, 如果我们选定的基准值就有相同的元素, 而基准值我们只选定了其中一个, 这样会降低排序的效率
  • 所以我们可以按照**"将数组分成三块"** 的思想实现快排, 在处理数据量有很多重复的情况下,效率会大大提升。
  • 找基准值时, 我们要选取随机数, 经过大量的计算, 这样效率最高, 那我们可以生成下标, 就相当于生成了数组中的随机数, 我们要生成的随机下标要在我们所排序的下标范围内方法时:
    int key = nums[new Random().nextInt(r - l + 1) + l];//l和r表示此时需要排序的左右位置
    • 快速排序算法主要流程:
      a. 定义递归出口;
      b. 利⽤随机选择基准函数⽣成⼀个基准元素;
      c. 利⽤荷兰国旗思想将数组划分成三个区域;
      d. 递归处理左边区域和右边区域。

代码:

java 复制代码
class Solution {
    public int[] sortArray(int[] nums) {
        qsort(nums, 0, nums.length - 1);
        return nums;
    }

    public void qsort(int[] nums, int l, int r) {
        if (l >= r)
            return;
        int key = nums[new Random().nextInt(r - l + 1) + l];
        int left = l - 1;
        int right = r + 1;
        int i = l;
        while (i < right) {
            if (nums[i] < key) {
                swap(nums, i++, ++left);
            } else if (nums[i] == key) {
                i++;
            } else {
                swap(nums, i, --right);
            }
        }
        qsort(nums, l, left);
        qsort(nums, right, r);
    }

    public void swap(int[] nums, int i, int j) {
        int tmp = nums[i];
        nums[i] = nums[j];
        nums[j] = tmp;
    }
}
相关推荐
山上三树1 分钟前
详细介绍 C 语言 typedef 及与 #define 的核心对比
c语言·数据结构·算法
释怀°Believe7 分钟前
Daily算法刷题【面试经典150题-7️⃣位运算/数学/】
算法·面试·职场和发展
2401_8762213417 分钟前
因数个数、因数和、因数积
c++·算法
云里雾里!27 分钟前
LeetCode 744. 寻找比目标字母大的最小字母 | 从低效到最优的二分解法优化
算法·leetcode
一条大祥脚42 分钟前
26.1.3 快速幂+容斥 树上dp+快速幂 带前缀和的快速幂 正序转倒序 子序列自动机 线段树维护滑窗
数据结构·算法
二狗哈1 小时前
czsc入门5: Tick RawBar(原始k线) NewBar (新K线)
算法·czsc
꧁Q༒ོγ꧂1 小时前
算法详解(四)--排序与离散化
数据结构·算法·排序算法
Tisfy1 小时前
LeetCode 0865.具有所有最深节点的最小子树:深度优先搜索(一次DFS + Python5行)
算法·leetcode·深度优先·dfs·题解
Q741_1471 小时前
C++ 队列 宽度优先搜索 BFS 力扣 429. N 叉树的层序遍历 C++ 每日一题
c++·算法·leetcode·bfs·宽度优先
Yzzz-F1 小时前
P4145 上帝造题的七分钟 2 / 花神游历各国[线段树 区间开方(剪枝) + 区间求和]
算法·机器学习·剪枝