(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】

这篇具有很好参考价值的文章主要介绍了(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

❓剑指 Offer 51. 数组中的逆序对

难度:困难

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数。

示例 1:

输入: [7,5,6,4]
输出: 5

限制

  • 0 <= 数组长度 <= 50000

💡思路:归并排序

预备知识

归并排序」是用分治思想,分治模式在每一层递归上有三个步骤:

  1. 分解(Divide):将 n 个元素分成个含 n/2 个元素的子序列。
  2. 解决(Conquer):用 合并排序法 对两个子序列递归的排序。
  3. 合并(Combine):合并两个已排序的子序列已得到排序结果。

(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】,LeetCode,leetcode,算法,职场和发展
具体的我们以一组无序数列{14,12,15,13,11,16}为例分解说明,如下图所示:

(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】,LeetCode,leetcode,算法,职场和发展

在待排序序列长度为 1 的时候,递归开始「回升」,因为我们默认长度为 1 的序列是排好序的。

具体思路

那么求逆序对和归并排序又有什么关系呢?关键就在于「归并」当中「」的过程。

合并阶段 本质上是 合并两个排序数组 的过程:

  • 每当遇到 左子数组当前元素 > 右子数组当前元素 时,意味着
    • 左子数组当前元素i 至 末尾元素m」 与 「右子数组当前元素」 构成了若干 「逆序对」 ;
    • 逆序对数 cnts += (m - i + 1)
  • 考虑在归并排序的合并阶段统计「逆序对」数量,完成归并排序时,也随之完成所有逆序对的统计。

🍁代码:(C++、Java)

C++

class Solution {
private:
    int cnts = 0;
    void mergeSort(vector<int>& nums, vector<int>& tmp, int l, int h){
        if(h - l < 1) return;
        //分解
        int m = l + (h - l) / 2;
        mergeSort(nums, tmp, l, m);
        mergeSort(nums, tmp, m + 1, h);
        //解决 + 合并
        int k = l, i = l, j = m + 1;
        while(i <= m || j <= h){
            if(i > m) tmp[k++] = nums[j++];
            else if(j > h || nums[i] <= nums[j]) tmp[k++] = nums[i++];
            else{//此时i~m对应数组中的数都比nums[j]大
                tmp[k++] = nums[j++];
                cnts += (m - i + 1);
            } 
        }
        for(k = l; k <= h; k++){
            nums[k] = tmp[k];
        }
    }
public:
    int reversePairs(vector<int>& nums) {
        vector<int> tmp(nums.size());//辅助数组,临时记录中间合并的子数组
        mergeSort(nums, tmp, 0, nums.size() - 1);
        return cnts;
    }
};

Java

class Solution {
    private int cnts = 0;
    private void mergeSort(int[] nums, int[] tmp, int l, int h){
        if(h - l < 1) return;
        //分解
        int m = l + (h - l) / 2;
        mergeSort(nums, tmp, l, m);
        mergeSort(nums, tmp, m + 1, h);
        //解决 + 合并
        int k = l, i = l, j = m + 1;
        while(i <= m || j <= h){
            if(i > m) tmp[k++] = nums[j++];
            else if(j > h || nums[i] <= nums[j]) tmp[k++] = nums[i++];
            else{//此时i~m对应数组中的数都比nums[j]大
                tmp[k++] = nums[j++];
                cnts += (m - i + 1);
            } 
        }
        for(k = l; k <= h; k++){
            nums[k] = tmp[k];
        }
    }
    public int reversePairs(int[] nums) {
        int[] tmp = new int[nums.length];//辅助数组,临时记录中间合并的子数组
        mergeSort(nums, tmp, 0, nums.length - 1);
        return cnts;
    }
}
🚀 运行结果:

(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】,LeetCode,leetcode,算法,职场和发展

🕔 复杂度分析:
  • 时间复杂度 O ( n l o g n ) O(nlogn) O(nlogn),其中 n 为数组的长度,同归并排序 O ( n l o g n ) O(nlogn) O(nlogn)
  • 空间复杂度 O ( n ) O(n) O(n),归并排序需要用到一个临时数组。

题目来源:力扣。

放弃一件事很容易,每天能坚持一件事一定很酷,一起每日一题吧!
关注我LeetCode主页 / CSDN—力扣专栏,每日更新!文章来源地址https://www.toymoban.com/news/detail-676412.html

注: 如有不足,欢迎指正!

到了这里,关于(排序) 剑指 Offer 51. 数组中的逆序对 ——【Leetcode每日一题】的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • (数组与矩阵) 剑指 Offer 03. 数组中重复的数字 ——【Leetcode每日一题】

    难度:简单 找出数组中重复的数字。 在一个长度为 n 的数组 nums 里的所有数字都在 0~n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。 示例 1: 输入 : [2, 3, 1, 0, 2, 5, 3] 输出 :2 或

    2024年02月16日
    浏览(39)
  • (链表) 剑指 Offer 25. 合并两个排序的链表 ——【Leetcode每日一题】

    难度:简单 输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然是递增排序的。 示例1: 输入:1-2-4, 1-3-4 输出:1-1-2-3-4-4 限制 : 0 = 链表长度 = 1000 注意:本题与 21. 合并两个有序链表 相同 💡思路: 法一:递归 将该问题可以分解成子链表,只比较当前 l1 链

    2024年02月15日
    浏览(45)
  • Leetcode-每日一题【剑指 Offer II 010. 和为 k 的子数组】

    给定一个整数数组和一个整数 k ,请找到该数组中和为 k 的连续子数组的个数。 示例 1: 输入: nums = [1,1,1], k = 2 输出: 2 解释: 此题 [1,1] 与 [1,1] 为两种不同的情况 示例 2: 输入: nums = [1,2,3], k = 3 输出: 2 提示: 1 = nums.length = 2 * 104 1000 = nums[i] = 1000 107 = k = 107 前置知识 前缀和

    2024年02月15日
    浏览(47)
  • (动态规划) 剑指 Offer 42. 连续子数组的最大和 ——【Leetcode每日一题】

    难度:简单 输入一个整型数组,数组中的一个或连续多个整数组成一个子数组。求所有子数组的和的最大值。 要求时间复杂度为 O(n) 。 示例1: 输入: nums = [-2,1,-3,4,-1,2,1,-5,4] 输出: 6 解释: 连续子数组 [4,-1,2,1] 的和最大,为 6。 提示 : 1 = a r r . l e n g t h = 1 0 5 1 = arr.length = 10^

    2024年02月11日
    浏览(53)
  • (搜索) 剑指 Offer 12. 矩阵中的路径 ——【Leetcode每日一题】

    难度:中等 给定一个 m * n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许

    2024年02月12日
    浏览(42)
  • 每天一道leetcode:剑指 Offer 53 - I. 在排序数组中查找数字 I(适合初学者&二分查找)

    统计一个数字在排序数组中出现的次数。 0 = nums.length = 10^5 -10^9 = nums[i] = 10^9 nums 是一个非递减数组 -10^9 = target = 10^9 使用两次二分查找找到target在数组中的左右边界,然后长度就是右边界减去左边界再加一,最后返回长度即可。   欢迎大家在评论区讨论,如有不懂的代码部分

    2024年02月14日
    浏览(57)
  • Leetcode 剑指 Offer II 038. 每日温度

    题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer(专项突击版)系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 请根据每日 气温 列表 temperatures ,重新生成一个列表,要求其对应位置的输出为:要想观测到更高的

    2024年02月14日
    浏览(46)
  • Leetcode-每日一题【剑指 Offer 29. 顺时针打印矩阵】

    输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。 示例 1: 输入: matrix = [[1,2,3],[4,5,6],[7,8,9]] 输出: [1,2,3,6,9,8,7,4,5] 示例 2: 输入: matrix = [[1,2,3,4],[5,6,7,8],[9,10,11,12]] 输出: [1,2,3,4,8,12,11,10,9,5,6,7] 限制: 0 = matrix.length = 100 0 = matrix[i].length = 100 1.题目要求

    2024年02月13日
    浏览(57)
  • (链表) 剑指 Offer 24. 反转链表 ——【Leetcode每日一题】

    难度:简单 定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点。 示例: 输入 : 1-2-3-4-5-NULL 输出 : 5-4-3-2-1-NULL 限制 : 0 = 节点个数 = 5000 注意:本题与 206. 反转链表 相同。 💡思路: 法一:递归 可以将本问题分解成子问题: 1 - (剩余部分的反转)

    2024年02月15日
    浏览(49)
  • Leetcode-每日一题【剑指 Offer 16. 数值的整数次方】

    实现 pow(x, n) ,即计算 x 的 n 次幂函数(即,xn)。不得使用库函数,同时不需要考虑大数问题。 示例 1: 输入: x = 2.00000, n = 10 输出: 1024.00000 示例 2: 输入: x = 2.10000, n = 3 输出: 9.26100 示例 3: 输入: x = 2.00000, n = -2 输出: 0.25000 解释: 2-2 = 1/22 = 1/4 = 0.25 提示: -10

    2024年02月13日
    浏览(41)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包