题解 | #数组中的逆序对#

数组中的逆序对

https://www.nowcoder.com/practice/96bd6684e04a44eb80e6a68efc0ec6c5?tpId=13&&tqId=11188&rp=1&ru=/ta/coding-interviews&qru=/ta/coding-interviews/question-ranking

35、数组中的逆序对

解题思路

题目简单易懂就不做过多解释,一开始很容易想到解题可以使用暴力法去统计所有的逆序对,但是这样的话时间复杂度是O(n2)

方法一: 暴力统计法

先直接给出暴力法的代码:

public class Solution {

    public int InversePairs(int[] array) {
        int cnt = 0;
        int len = array.length;
        for (int i = 0; i < len - 1; i++) {
            for (int j = i + 1; j < len; j++) {
                if (array[i] > array[j]) {
                    cnt++;
                }
            }
        }
        return cnt;
    }
}

我们可以看到使用了两个for循环,所以时间复杂度是很高的!这在面试中必是不被允许的!

方法二:归并统计法

相信大家都知道归并排序这个算法吧,如果不知道的,可以移至这里排序先去学习一波~

那么,我们先来说说归并算法吧,归并算法讲究一个先分后并!

先分:分呢,就是将数组分为两个子数组,两个子数组分为四个子数组,依次向下分,直到数组不能再分为止!

后并:并呢,就是从最小的数组按照顺序合并,从小到大或从大到小,依次向上合并,最后得到合并完的顺序数组!

介绍完归并排序,我们来说说归并统计法,我们要在哪个步骤去进行统计呢?

归并统计法,关键点在于合并环节,在合并数组的时候,当发现右边的小于左边的时候,此时可以直接求出当前产生的逆序对的个数。

举个例子:

在合并 {4 ,5} {1 , 2} 的时候,首先我们判断 1 < 4,我们即可统计出逆序对为2,为什么呢?这利用了数组的部分有序性。因为我们知道 {4 ,5} 这个数组必然是有序的,因为是合并上来的。此时当 1比4小的时候,证明4以后的数也都比1大,此时就构成了从4开始到 {4,5}这个数组结束,这么多个逆序对(2个),此时利用一个临时数组,将1存放起来,接着比较2和4的大小,同样可以得到有2个逆序对,于是将2也放进临时数组中,此时右边数组已经完全没有元素了,则将左边剩余的元素全部放进临时元素中,最后将临时数组中的元素放进原数组对应的位置。

最后接着向上合并~

可以看到下面这张图~

image-20210623223031128

下面直接对着代码进行解释:

public class Solution {
    int count = 0;
    public int InversePairs(int [] array) {
        // 长度小于2则无逆序对
        if(array.length < 2)
            return 0;
        // 进入归并
        mergeSort(array,0,array.length-1);
        return count;
    }

    public void mergeSort(int[] array,int left,int right){
        // 找分割点
        int mid = left+(right-left)/2;
        if(left < right){
            // 左子数组
            mergeSort(array,left,mid);
            // 右子数组
            mergeSort(array,mid+1,right);
            // 并
            merge(array,left,mid,right);
        }
    }

    public void merge(int[] array,int left,int mid,int right){
        // 创建临时数组,长度为此时两个子数组加起来的长度
        int[] arr =  new int[right-left+1];
        // 临时数组的下标起点
        int c = 0;
        // 保存在原数组的起点下标值
        int s = left;
        // 左子数组的起始指针
        int l = left;
        // 右子数组的起始指针
        int r = mid+1;
        while(l <= mid && r <= right ){
            // 当左子数组的当前元素小的时候,跳过,无逆序对
            if(array[l] <= array[r]){
                // 放入临时数组
                arr[c] = array[l];
                // 临时数组下标+1
                c++;
                // 左子数组指针右移
                l++;
            }else{ // 否则,此时存在逆序对
                // 放入临时数组
                arr[c] = array[r];
                // 逆序对的个数为    左子数组的终点- 当前左子数组的当前指针
                count += mid+1-l;
                count %= 1000000007;
                // 临时数组+1
                c++;
                // 右子数组的指针右移
                r++;
            }
        }

        // 左子数组还有元素时,全部放入临时数组
        while(l <= mid)
            arr[c++] = array[l++];
        // 右子数组还有元素时,全部放入临时数组
        while(r <= right)
            arr[c++] = array[r++];
        // 将临时数组中的元素放入到原数组的指定位置
        for(int num:arr){
            array[s++] = num;
        }
    }
}

复杂度分析:

时间复杂度:O(NlogN)。归并排序的时间复杂度(建议百度~)

空间复杂度:O(N)。临时数组的空间。

剑指offer 文章被收录于专栏

为刷过的每一道题都书写一篇题解,便于重复练习~

全部评论
```go var count int func InversePairs(nums []int) int { count = 0 temp := make([]int, len(nums)) mergeSort(nums, temp, 0, len(nums)-1) return count } func insertSort(nums []int, left, right int) { for i := left + 1; i <= right; i++ { t := nums[i] j := i for j > left && t < nums[j-1] { count++ nums[j] = nums[j-1] j-- } nums[j] = t } } func mergeSort(nums, temp []int, left, right int) { if right - left + 1 < 10 { insertSort(nums, left, right) return } mid := (left + right) / 2 mergeSort(nums, temp, left, mid) mergeSort(nums, temp, mid+1, right) if nums[mid] <= nums[mid+1] { return } merge(nums, temp, left, mid, right) } func merge(nums, temp []int, left, mid, right int) { copy(temp, nums) i := left j := mid+1 for k := left; k <= right; k++ { if i > mid { nums[k] = temp[j] j++ } else if j > right { nums[k] = temp[i] i++ } else if temp[i] <= temp[j] { nums[k] = temp[i] i++ } else { nums[k] = temp[j] count += mid+1 - i count %= 1000000007 j++ } } } ```
1
送花
回复
分享
发布于 2022-10-06 17:18 四川
大佬 count求余我把它放在了最后求余,这样得出来的是个负数 这是为什么啊 放在while循环里和放在最后有什么区别吗
点赞
送花
回复
分享
发布于 2021-08-25 20:22
秋招专场
校招火热招聘中
官网直投
while循环里用了&&,后续还会有左边或右边多余吗、???
点赞
送花
回复
分享
发布于 2022-06-21 09:04
讲得好,但我还是选择用暴力解法
点赞
送花
回复
分享
发布于 02-04 10:16 陕西

相关推荐

点赞 评论 收藏
转发
105 27 评论
分享
牛客网
牛客企业服务