参考:1、 https://www.geeksforgeeks.org/merge-sort/

2、《剑指Offer:名企面试官精讲典型编程题》

题目描述

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。 即输出P%1000000007

输入描述:

题目保证输入的数组中没有的相同的数字

数据范围:

  • 对于%50的数据,size<=10^4
  • 对于%75的数据,size<=10^5
  • 对于%100的数据,size<=2*10^5
示例1
输入:1,2,3,4,5,6,7,0
输出:7

解题思路1 (暴力法,时间复杂度: O(n^2))

这个方法最简单直接,顺序扫描整个数组,每扫到一个数,逐个比较其后面的所有数,如果构成逆序对,则计数+1。代码没啥难度,略了。

解题思路2 (分治法,时间复杂度: O(nlogn),空间复杂度:O(n))

     public int InversePairs(int[] array) {
         if (array.length < 2) {
             return 0;
         }
         return divideAndConquer(array, 0, array.length);
     }

     public int divideAndConquer(int[] array, int l, int r) {
         if (l >= r - 1) return 0;

         int mid = (l + r) / 2;
         int count = 0;
         int countL = divideAndConquer(array, l, mid);
         int countR = divideAndConquer(array, mid, r);
         int[] left = Arrays.copyOfRange(array, l, mid);
         int[] right = Arrays.copyOfRange(array, mid, r);
         int i = left.length - 1;
         int j = right.length - 1;
         for (int k = r - 1; k >= l; k--) {
             // two cases: left is empty or right is empty
             if (i < 0) {
                 array[k] = right[j];
                 j--;
                 continue;
             } else if (j < 0) {
                 array[k] = left[i];
                 i--;
                 continue;
             }
             if (left[i] > right[j]) {
                 count += j + 1;
                 if (count > 1000000007) {
                     count %= 1000000007;
                 }
                 array[k] = left[i];
                 i--;
             } else {
                 array[k] = right[j];
                 j--;
             }
         }
         return (count + countL + countR) % 1000000007;
     }

这一题的解法是在归并排序(Merge Sort) 的基础上修改得到的。Merge Sort使用的是分治法的思想。分治法 (divide and conquer) 是将一个复杂的问题,分成两个或多个相同或者相似的子问题,再把子问题分成更小的子问题,一直分到子问题足够简单来求解,最后再把子问题的解合并成原问题的解。归并排序法的时间复杂度为 O(nlogn),空间复杂度为 临时的数组 + 递归时压入栈的数据占用的空间 = n + logn,即 O(n)。

以数组 {7, 5, 6, 4} 为例进行分析,解题过程如图5.2所示。首先将数组拆成至长度为1的子数组,然后一边合并一边统计逆序对的数目。在第一对长度为1 的子数组 {7}、{5}中,有一对逆序对 (7, 5)。在第二对长度为1的子数组 {6}、{4}中,也有一对逆序对(6, 4)。在统计完这两对逆序对后,需要分别对他们进行排序,如图 5.2(c),以免在以后的统计过程中再重复统计。

然后需要统计两个长度为2的子数组之间的逆序对,并合并两个子数组,如图5.3所示。两个子数组分别从末尾开始判断数的大小,如图5.2(a)中指针P1、P2。如果 P1 大于 P2,则构成逆序对,并且逆序对的数目等于第二个子数组中剩余数字的个数,即 P2 以及之前的所有数字的个数。如果 P1 小于或等于 P2, 则不构成逆序对。每次比较的时候,我们都把较大的数字从后往前复制到一个辅助数组,确保辅助数组中的数字是递增排序的。在把较大的数字复制到辅助数组之后,把对应的指针向前移动一位,接下来进行下一轮比较。

以此类推,使用这个方法去求出其他数组的逆序对个数。在解这一题的时候,还有一点需要注意,除了需要将总输出结果 (count + countL + countR) 对1000000007取模,还要在计算过程中将 count += j + 1 对1000000007取模,以免count超出 Integer.MAX_VALUE。

剑指offer-36:数组中的逆序对的更多相关文章

  1. 剑指 Offer 51. 数组中的逆序对 + 归并排序 + 树状数组

    剑指 Offer 51. 数组中的逆序对 Offer_51 题目描述 方法一:暴力法(双层循环,超时) package com.walegarrett.offer; /** * @Author Wal ...

  2. 【Java】 剑指offer(51)数组中的逆序对

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 在数组中的两个数字如果前面一个数字大于后面的数字,则这两个数字组成 ...

  3. 【剑指offer】数组中的逆序对

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/mmc_maodun/article/details/27520535 转载请注明出处:http:// ...

  4. Go语言实现:【剑指offer】数组中的逆序对

    该题目来源于牛客网<剑指offer>专题. 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对10000 ...

  5. 【剑指offer】数组中的逆序对。C++实现

    原创文章,转载请注明出处! 博客文章索引地址 博客文章中代码的github地址 # 题目 # 思路 基于归并排序的思想统计逆序对:先把数组分割成子数组,再子数组合并的过程中统计逆序对的数目.统计逆序对 ...

  6. 微软面试题:剑指 Offer 51. 数组中的逆序对 Hard 出现次数:3

    题目描述: 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对. 输入一个数组,求出这个数组中的逆序对的总数. 示例 1: 输入: [7,5,6,4] 输出: 5 限制: ...

  7. 剑指Offer 35. 数组中的逆序对 (数组)

    题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即输出P%1000 ...

  8. [剑指Offer] 35.数组中的逆序对

    题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即输出P%1000 ...

  9. 剑指offer:数组中的逆序对

    题目描述: 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即输出P%100 ...

  10. 剑指offer——54数组中的逆序对

    题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数P.并将P对1000000007取模的结果输出. 即输出P%1000 ...

随机推荐

  1. 是谁,在敲打我窗-CSS雨滴动画效果

    1.扯闲篇 是谁在敲打我窗  是谁在撩动琴弦  那一段被遗忘的时光 渐渐地回升出我心坎  是谁在敲打我窗  是谁在撩动琴弦 记忆中那欢乐的情景  慢慢地浮现在我的脑海 那缓缓飘落的小雨  不停地打在我 ...

  2. 【Android - 自定义View】之自定义颜色渐变的Tab导航栏

    首先来介绍一下这个自定义View: (1)这个自定义View的名称叫做 GradientTab ,继承自View类: (2)这个自定义View实现了颜色渐变的Tab导航栏(仿微信主菜单),用户在左右滑 ...

  3. Chapter 07-Basic statistics(Part1 描述统计数据)

    在这一部分中,仍然使用mtcars(Motor Trend Car Road Tests)这一数据集,以及mpg(one mile per gallon), hp(horsepower), wt(we ...

  4. 安装破解版IntelliJ IDEA

    1.下载IntelliJ IDEA http://www.jetbrains.com/idea/download/#section=windows 选择Ultimate版本 2.注册码破解 http: ...

  5. 洛谷 题解 P5535 【【XR-3】小道消息】

    我又双叒叕被包菜辣! P5535 [XR-3]小道消息(这道题是个大水题 在题干中这位良心的作者就提醒了我们: 你可能需要用到的定理--伯特兰-切比雪夫定理. 那么什么是伯特兰-切比雪夫定理? 我也不 ...

  6. 洛谷 题解 CF1151D 【Stas and the Queue at the Buffet】

    本蒟蒻又双叒叕被爆踩辣!!! 题目链接 这道题我个人觉得没有紫题的水平. 步入正题 先看题: 共有n个人,每个人2个属性,a,b; 窝们要求的是总的不满意度最小,最满意度的公式是什么? \(ai * ...

  7. gsoap使用

    一. 安装gsoap 下载地址:http://sourceforge.net/projects/gsoap2/files/ 解压安装:./configure --prefix=/usr/local/g ...

  8. IPhone下json的解析 NSJSONSerialization

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式. 易于人阅读和编写.同时也易于机器解析和生成. 它基于JavaScript Programming Lan ...

  9. [TimLinux] 开博一个月了

    做事情,怕的确实是坚持!为了自己尽可能的每天一篇博文,时常在下班后,23点开始写博,有时候写到接近一点钟.但是这第一个月,确实坚持下来了.平均每天一篇.写博的过程,其实是: 知识的总结过程: 因为要求 ...

  10. nitacm20317 来自张司机的挑战书

    题目:让你求从x到y中(1<=x<=y<=10^18),二进制一的个数最多的数是哪个,如果有多个相同的答案,输出最小的. 题目链接:https://www.nitacm.com/pr ...