码迷,mamicode.com
首页 > 其他好文 > 详细

lintcode:逆序对

时间:2016-07-07 13:05:52      阅读:139      评论:0      收藏:0      [点我收藏+]

标签:

题目

在数组中的两个数字如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。给你一个数组,求出这个数组中逆序对的总数。
概括:如果a[i] > a[j] 且 i < j, a[i] 和 a[j] 构成一个逆序对。

样例

序列 [2, 4, 1, 3, 5] 中,有 3 个逆序对 (2, 1)(4, 1)(4, 3),则返回 3 。

解题

直接暴力找,时间复杂度O(n^2)

public class Solution {
    /**
     * @param A an array
     * @return total of reverse pairs
     */
    public long reversePairs(int[] A) {
        // Write your code here
        long res = 0;
        int n = A.length;
        for(int i=0;i<n;i++){
            for(int j=i+1;j<n;j++){
                if(A[i] >A[j]){
                    res ++;
                }
            }
        }
        return res;
    }
}

归并排序的思想

对于数组A[p,...,q]

分成两个数组A[p,...,r],A[r+1,...,q]

当这两个数组都是有序的时候,其逆序对数很好求

可以两个指针i,j分布指向数组的尾部

当A[i]>A[j]时候一定是逆序对,注意是有序的,逆序对数量:j-(r+1) +1 = j-r

else 不是逆序对

程序

public class Solution {
    /**
     * @param A an array
     * @return total of reverse pairs
     */
     long res = 0;
    public long reversePairs(int[] A) {
        // Write your code here
        
        int n = A.length;
        reversP(A,0,n-1);
        return res;
    }
    public void reversP(int[] A,int low,int high){
        if(low>=high)
            return;
        int mid = low + (high - low)/2;
        reversP(A,low,mid);
        reversP(A,mid+1,high);
        merge(A,low,mid,high);
    }
    public void merge(int[] A,int low ,int mid ,int high){
        int len = high - low + 1; 
        int[] C = new int[len]; // 临时存放中间归并数组
        int i = mid;
        int j = high;
        int k = len -1;
        while(i>= low && j>=mid+1){
            if(A[i] > A[j]){
                C[k--] = A[i];
                i--;
                res += j - (mid + 1) +1; // 逆序对数量
            }else{
                C[k--] = A[j];
                j--;
            }
        }
        while(i>=low){
            C[k--] = A[i];
            i--;
        }
        while(j>=mid+1){
            C[k--] = A[j];
            j--;
        }

        for(k=0;k<len;k++){
            A[k+low] = C[k];
        }

    }
    public void print(int[] A){
        for(int a:A){
            System.out.print(a+"\t");
        }
        System.out.println();
    }
}

分析下输出过程

测试样例:[2,4,1,3,5]

输出情况

2   4   1   3   5   
1   2   4   3   5   
1   2   4   3   5   
1   2   3   4   5 
 

lintcode:逆序对

标签:

原文地址:http://www.cnblogs.com/theskulls/p/5649519.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!