剑指offer51-数组中的逆序对

原题(来源Leetcode)

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

示例 1:

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

限制:

0 <= 数组长度 <= 50000

解法

思想

暴力:双重循环检查,leetcode中超时

分治思想:使用归并排序的算法,每次给左右两个数组合并的时候,若右边数组中指向的元素小于左边数组指向的元素,则也小于左边数组指向的元素后面的任何一个元素(归并排序合并的时候左右两个数组都是有序的),则此时加上左边数组剩余的元素个数即可。

代码

归并排序大复习,这里提供两种写法:

下面是归并排序较简单的写法,就是每次将数组均分成两个子数组,分到不能再分的时候开始合并相邻的两个子数组。这种写法递归中会创建新数组并拷贝,因此时间成本会略高些。

class Solution {
    int count;//全局变量计数
    public int reversePairs(int[] nums) {
        MergeSort(nums);
        return count;
    }
    public int[] MergeSort(int[] array) {
        if (array.length < 2) return array;
        int mid = array.length / 2;
        int[] left = Arrays.copyOfRange(array, 0, mid);
        int[] right = Arrays.copyOfRange(array, mid, array.length);
        return merge(MergeSort(left), MergeSort(right));
    }
    public int[] merge(int[] left, int[] right) {
        int[] result = new int[left.length + right.length];
        for (int index = 0, i = 0, j = 0; index < result.length; index++) {
            if (i == left.length)
                result[index] = right[j++];
            else if (j == right.length)
                result[index] = left[i++];
            else if (left[i] > right[j]){
                count+=left.length-i;//加上左边数组剩余的元素个数
                result[index] = right[j++];
            }
            else result[index] = left[i++];
        }
        return result;
    }
}

下面是一种递归过程中不用创建数组的归并排序写法,只使用两个和原数组相同大小的数组来回交换数据(做对方的缓存),在局部上操作,时间成本会比上面创建数组的方法更低些,但是是一个量级的,在leetcode中能取得更好的成绩:

class Solution {
    public int reversePairs(int[] nums) {
        int[] temp = nums.clone();//创建一个和原数组相同大小的数组
        return MergeSort(nums, temp, 0, nums.length - 1);
    }

    //[start,end]确定一个数组范围,即递归中要排序的范围,返回值为合并过程中的计数
    private int MergeSort(int[] nums, int[] temp, int start, int end) {
        if (start >= end) return 0;
        int mid = (start + end)/2;
        int left = MergeSort(temp, nums, start, mid);
        int right = MergeSort(temp, nums, mid + 1, end);

        int count = merge(nums, temp, start, mid+1, end);
        return left + right + count;
    }

    //[left,right)和[right,end]确定数组的两部分,合并temp数组中的这两部分到nums数组
    private int merge(int[] nums, int[] temp, int left, int right, int end) {
        int index = left;
        int leftSize = right;//因为直接使用left和right作为后面使用的双指针,这里保存right原位置作为left指针能移动到的最远处
        int count = 0;//合并过程中计数

        while (index <= end) {
            if (left==leftSize) nums[index] = temp[right++];
            else if (right==end+1) nums[index] = temp[left++];
            else if (temp[left] > temp[right]) {
                count += (leftSize - left);//加上左边部分剩余的元素个数
                nums[index] = temp[right++];
            } else nums[index] = temp[left++];
            index++;
        }

        return count;
    }
}

原创文章,作者:彭晨涛,如若转载,请注明出处:https://www.codetool.top/article/%e5%89%91%e6%8c%87offer51-%e6%95%b0%e7%bb%84%e4%b8%ad%e7%9a%84%e9%80%86%e5%ba%8f%e5%af%b9/

(0)
彭晨涛彭晨涛管理者
上一篇 2020年4月23日
下一篇 2020年4月24日

相关推荐

  • leetcode530-二叉搜索树的最小绝对差

    原题 给定一个所有节点为非负值的二叉搜索树,求树中任意两节点的差的绝对值的最小值。 示例: 输入:    1    &n…

    算法 2020年2月22日
    01330
  • leetcode160-相交链表

    原题 编写一个程序,找到两个单链表相交的起始节点。 如下面的两个链表: 在节点 c1 开始相交。 示例 1: 输入: intersectVal = 8, listA = [4,1,…

    2019年12月14日
    090
  • leetcode104-二叉树的最大深度

    原题 给定一个二叉树,找出其最大深度。 二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。 说明: 叶子节点是指没有子节点的节点。 示例:给定二叉树 [3,9,20,null…

    算法 2020年1月11日
    090
  • leetcode1095-山脉数组中查找目标值

    原题 (这是一个 交互式问题 ) 给你一个 山脉数组 mountainArr,请你返回能够使得 mountainArr.get(index) 等于 target 最小 的下标 in…

    算法 2020年4月29日
    0160
  • leetcode32-最长有效括号

    原题 给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 示例 1: 输入: "(()" 输出: 2 解释: 最长有效括号子串为 "()" 示例 2…

    算法 2020年6月12日
    0160
  • leetcode739-每日温度

    原题 根据每日 气温 列表,请重新生成一个列表,对应位置的输入是你需要再等待多久温度才会升高超过该日的天数。如果之后都不会升高,请在该位置用 0 来代替。 例如,给定一个列表 te…

    算法 2019年12月11日
    0130
  • leetcode215-数组中的第K个最大元素

    原题 在未排序的数组中找到第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。 示例1: 输入: [3,2,1,5,6,4] 和…

    算法 2020年2月10日
    0140
  • leetcode38-外观数列

    原题 「外观数列」是一个整数序列,从数字 1 开始,序列中的每一项都是对前一项的描述。前五项如下: 1. 1 2. 11 3. 21 4. 1211 5. 111221 1 被读作…

    算法 2020年4月30日
    0290
  • leetcode560-和为K的子数组

    原题 给定一个整数数组和一个整数 k,你需要找到该数组中和为 k 的连续的子数组的个数。 示例 1: 输入: nums = [1,1,1], k = 2 输出: 2 , [1,1]…

    算法 2020年5月15日
    0470
  • 剑指offer35-复杂链表的复制

    原题(来源Leetcode) 请实现 copyRandomList 函数,复制一个复杂链表。在复杂链表中,每个节点除了有一个 next 指针指向下一个节点,还有一个 random …

    2020年6月13日
    0110

发表回复

登录后才能评论