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也放进临时数组中,此时右边数组已经完全没有元素了,则将左边剩余的元素全部放进临时元素中,最后将临时数组中的元素放进原数组对应的位置。
最后接着向上合并~
可以看到下面这张图~
下面直接对着代码进行解释:
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)。临时数组的空间。