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

转载请注明出处:http://blog.csdn.net/ns_code/article/details/27520535

题目描写叙述:
在数组中的两个数字。假设前面一个数字大于后面的数字,则这两个数字组成一个逆序对。

输入一个数组。求出这个数组中的逆序对的总数。

输入:
每一个測试案例包括两行:
第一行包括一个整数n,表示数组中的元素个数。

当中1 <= n <= 10^5。

第二行包括n个整数,每一个数组均为int类型。
输出:
相应每一个測试案例。输出一个整数,表示数组中的逆序对的总数。
例子输入:
4
7 5 6 4
例子输出:
5
    思路:最简单的方法是顺序数组。将每一个数字与后面的比較,统计逆序对的个数,这样的方法的时间复杂度为O(n*n),这样的方法写出的代码在九度OJ上測试,会超时。

剑指offer给出了归并排序的思路。这个有点难想到啊,也可能是我太弱了,根本没往这方面想!理解了思路。就不难了,将数组划分成两个子数组。再将子数组分别划分成两个子数组。统计每一个子数组内的逆序对个数,并将其归并排序。再统计两个子数组之间的逆序对个数,并进行归并排序。

这就是归并排序的变种,在归并排序代码的基础上稍作改进就可以。

    合理还要注意一点:全局变量count不能声明为int型。必须为long long型。由于题目中说数组最大为10^5,那么最大逆序对为(10^5-1)*10^5/2,这个数大约在50亿左右,超过了int型的表示范围。
    AC代码例如以下:
#include<stdio.h>
#include<stdlib.h> /*
统计两个子数组之间的逆序对
*/
long long MergePairsBetweenArray(int *arr,int *brr,int start,int mid,int end)
{
int i = mid;
int j = end;
int k = end; //辅助数组的最后一位
long long count = 0; //设置两个指针i,j分别从右往左依次比較。
//将较大的依次放入辅助数组的右边
while(i>=start && j>=mid+1)
{
if(arr[i] > arr[j])
{
count += j-mid;
brr[k--] = arr[i--];
}
else
brr[k--] = arr[j--];
} //将当中一个数组中还剩下的元素复制到辅助数组中。
//两个循环仅仅会运行当中的一个
while(i>=start)
brr[k--] = arr[i--];
while(j>=mid+1)
brr[k--] = arr[j--]; //从辅助数组中将元素复制到原数组中。使其有序排列
for(i=end;i>k;i--)
arr[i] = brr[i]; return count;
} /*
统计数组中的全部的逆序对
*/
long long CountMergePairs(int *arr,int *brr,int start,int end)
{
long long PairsNum = 0;
if(start<end)
{
int mid = (start+end)>>1;
PairsNum += CountMergePairs(arr,brr,start,mid); //统计左边子数组的逆序对
PairsNum += CountMergePairs(arr,brr,mid+1,end); //统计右边子数组的逆序对
PairsNum += MergePairsBetweenArray(arr,brr,start,mid,end); //统计左右子数组间的逆序对
}
return PairsNum;
} /*
将函数封装起来
*/
long long CountTotalPairs(int *arr,int len)
{
if(arr==NULL || len<2)
return 0; int *brr = (int *)malloc(len*sizeof(int));
if(brr == NULL)
exit(EXIT_FAILURE); long long sum = CountMergePairs(arr,brr,0,len-1);
free(brr);
brr = NULL; return sum;
} int main()
{
int n;
while(scanf("%d",&n) != EOF)
{
int *arr = (int *)malloc(n*sizeof(int));
if(arr == NULL)
exit(EXIT_FAILURE); int i;
for(i=0;i<n;i++)
scanf("%d",arr+i); printf("%lld\n",CountTotalPairs(arr,n)); free(arr);
arr = NULL;
}
return 0;
}
/**************************************************************
    Problem: 1348
    User: mmc_maodun
    Language: C
    Result: Accepted
    Time:100 ms
    Memory:1696 kb
****************************************************************/

【剑指offer】数组中的逆序对的更多相关文章

  1. [剑指OFFER] 数组中的逆序对

    题目描述 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数.     分析:利用归并排序的思想,分成2部分,每一部分按照从大到 ...

  2. 剑指Offer——数组中的逆序对

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

  3. 用js刷剑指offer(数组中的逆序对)

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

  4. 剑指Offer——数组中的逆序对(归并排序的应用)

    蛮力: 遍历数组,对每个元素都往前遍历所有元素,如果有发现比它小的元素,就count++. 最后返回count取模. 结果没问题,但超时哈哈哈,只能过50%.   归并法: 看讨论,知道了这道题的经典 ...

  5. 剑指 Offer——数组中的逆序对

    1. 题目 2. 解答 借助于归并排序的分治思想,在每次合并的时候统计逆序对.因为要合并的两个数组都是有序的,如果左半部分数组当前值大于右半部分数组当前值,那么左半部分数组当前值右边的数就都大于右半部 ...

  6. 剑指offer_数组中的逆序对

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

  7. 剑指Offer-34.数组中的逆序对(C++/Java)

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

  8. 剑指Offer34 数组中的逆序对

    /************************************************************************* > File Name: 34_Invers ...

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

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

  10. 剑指 Offer —— 数组中重复的数字

    数组中的重复数字 题目描述 牛课网链接 长度为 n 的数组里,所有数字都在 0 到 n-1 的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重复几次.请找出数组中任意一 ...

随机推荐

  1. FZU 2138——久违的月赛之一——————【贪心】

    久违的月赛之一 Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit Stat ...

  2. WebDriverWait介绍

    转自:https://www.cnblogs.com/ella-yao/p/7778678.html WebDriverWait介绍 package com.test.elementwait; imp ...

  3. DIY了一下自己blog的UI

    当年才学前端时就想改自己blog的UI,然鹅当时没看见那个“申请JS权限”,一直以为blog不能随意DIY样式,只改了少许CSS.现在重新看看设置管理选项,简单修改了一下样式(注意:修改样式之前发邮件 ...

  4. table中列复选框全选,再选 效果

    <table class="table table-striped sortable table-bordered table-hover " id="zdnews ...

  5. Why Isn't curr_items Decreasing When Items Expire?

    Why Isn't curr_items Decreasing When Items Expire?

  6. C#语言-04.OOP基础

    a. OOP:面对对象思想 i. 类:是一种功能强大的数据类型,而且是面向对象的基础 . 语法:访问修饰符 class 类名{ //类的主体 } . 成员变量:不以“函数”形式体现 a. 常量:代表与 ...

  7. vscode设置中文,设置中文不成功问题

    刚安装好的vscode界面显示英文,如何设置中文呢? 在locale.json界面设置”locale":"zh-cn"也未能实现界面为中文,在网上找了参考了,以下教程真实 ...

  8. java 错误: 未报告的异常错误Exception; 必须对其进行捕获或声明以便抛出

    import java.io.FileInputStream; import java.util.Properties; import java.sql.Connection; import java ...

  9. Effective C++ .12 复制对象-拷贝构造函数的编写

    当我们自己编写拷贝构造函数时,编译器就不会为该类生成默认拷贝构造函数了,对于assignment operator也是如此. 1. 拷贝构造函数中记得调用父类的拷贝构造函数,或者相应复制过程 clas ...

  10. es6 class类实例、静态、私有方法属性笔记

    实例属性.方法 class Foo { valueA = 100 //第一种实例属性定义,位置:new的实例上 constructor() { this.valueB = 200 //第二种实例属性定 ...