1 题目描述

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

2 思路和方法

  利用归并排序的思想,先把数组分隔成子数组,先统计出子数组内部的逆序对的数目,然后再统计出两个相邻子数组之间的逆序对的数目。注意在合并两个已排序的子数组后,要更新数组。O(n*log(n))。

3 C++核心代码

 class Solution {
public:
int InversePairs(vector<int> data) {
if(data.size() <= )
return ;
int count = ;
vector<int> copy(data); // 初始化
InversePairsCore(data, copy, , data.size()-,count);
return count;
} // 归并
void InversePairsCore(vector<int> &data, vector<int> &copy, int start, int end, int &count){
if(start>=end){
return;
}
int mid = (start + end) /;
InversePairsCore(data,copy,start,mid,count);
InversePairsCore(data,copy,mid+,end,count); int copyIndex = end; // 将较大数字从后往前复制到辅助数组
int i = mid; // 前半段最后一个元素下标
int j = end; // 后半段最后一个元素下标
while (i>=start && j>=mid+) {
if (data[j] < data[i]){ // 逆序
count += j - mid;
count %= ; // 取模防止逆序对溢出
copy[copyIndex--] = data[i--];
} else{
copy[copyIndex--] = data[j--];
}
} while (i>=start)
copy[copyIndex--] = data[i--]; while (j>=mid+)
copy[copyIndex--] = data[j--]; for (int k = start; k <= end; ++k)
data[k] = copy[k]; }
};

4 C++完整代码

 #include<iostream>
using namespace std;
//数组中的逆序对
long long GetMergePairsBetween(int* arr,int* copy,int start,int mid,int end)
{
//合并两个子数组,并计算逆序对个数
int final1 = mid;//第一个数组的最后一位
int final2 = end;//第二个数组的最后一位
int index = end;//辅助数组的最后一位
long long count = ;
while(final1 >= start && final2 >= mid+)//两个数组都没有处理完
{
if(arr[final1] > arr[final2])
{
//如果第一个数组的元素大于第二个数组的任何元素,
//则第一个数组的元素一定大于第个数组中final2之前的所有元素
count += (final2 - mid);
//将final1处的元素拷贝至copy数组
//index和final1都向前移动
copy[index--] = arr[final1--];
}
else
{
//第一个数组的元素小于第二个数组的元素
//第二个数组的元素拷贝至copy数组
//并将index和final2前移
copy[index--] = arr[final2--];
}
}
while(final1 >= start)//第一个数组的元素没有处理完
{
copy[index--] = arr[final1--];
}
while(final2 >= mid + )//第一个数组的元素没有处理完
{
copy[index--] = arr[final2--];
}
for(int i = end; i > index;i--)
arr[i] = copy[i];
return count;
}
long long GetMergePairs(int* arr,int* copy,int start,int end)
{
long long ret = ;
if(start < end)
{
int mid = start + ((end - start)>>);
ret += GetMergePairs(arr,copy,start,mid);
ret += GetMergePairs(arr,copy,mid+,end);
ret += GetMergePairsBetween(arr,copy,start,mid,end);
}
return ret;
}
long long GetTotalPairs(int arr[],int n)
{
if(arr == NULL || n < )
return ;
int* copy = new int[n];
long long sum = GetMergePairs(arr,copy,,n-);
delete[] copy;
return sum;
}
int main()
{
int arr[] = {,,,};
int ret = GetTotalPairs(arr,sizeof(arr)/sizeof(arr[]));
cout<<ret<<endl;
system("pause");
return ;
}

参考资料

https://blog.csdn.net/zjwreal/article/details/88769617

https://blog.csdn.net/DERRANTCM/article/details/46761051(图)

https://blog.csdn.net/peiyao456/article/details/54645952(完整代码)

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  10. 剑指offer--35.数组中只出现一次的数字

    时间限制:1秒 空间限制:32768K 热度指数:198150 本题知识点: 数组 题目描述 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. class ...

随机推荐

  1. 7月清北学堂培训 Day 5

    今天是钟皓曦老师的讲授~ 动态规划 动态规划的三种实现方法: 1.递推: 2.递归: 3.记忆化: 举个例子: 斐波那契数列:0,1,1,2,3,5,8…… Fn = Fn-1 + Fn-2 1.我们 ...

  2. idea将项目打成jar包

    在用jmeter做压测时,需要将项目打成jar包放至在如下目录 /Users/admin/Documents/software/apache-jmeter-5.1.1/apache-jmeter-5. ...

  3. varnish web cache服务

    varnish介绍 缓存开源解决方案: - varnish - 充分利用epoll机制(能显著提高程序在大量并发连接中只有少量活跃的情况下的系统CPU利用率),并发量大,单连接资源较轻 - squid ...

  4. java课后实验性问题2

    课后作业一:计算组合数 程序设计思想: 从键盘获取组合数,判断是否构成组合数.分别用三种方法计算组合数输出. 程序流程图: import java.util.Scanner; public class ...

  5. CDH 更换 HDFS 数据目录

    先停止 HDFS 角色. 数据文件位置默认在 /dfs/ 中,这里配置 NameNode.SecondaryNameNode.DataNode 数据目录. 先在所有 HDFS 的主机上把数据拷贝过去, ...

  6. 错误注入 异常行为 环境变量或代码动态激活来触发这些异常行为 模拟错误 容错性 正确性 稳定性 宏 本质 macro

    小结: 1. 微服务中某个服务出现随机延迟.某个服务不可用. 存储系统磁盘 IO 延迟增加.IO 吞吐量过低.落盘时间长. 调度系统中出现热点,某个调度指令失败. 充值系统中模拟第三方重复请求充值成功 ...

  7. C++ STL partial_sort_copy iterator

    vector<int>::iterator iter1 = partial_sort_copy(deq1.begin(), deq1.end(), vec1.begin(), vec1.e ...

  8. scikit-learn机器学习(三)多项式回归(二阶,三阶,九阶)

    我们仍然使用披萨直径的价格的数据 import matplotlib matplotlib.rcParams['font.sans-serif']=[u'simHei'] matplotlib.rcP ...

  9. 修改ssh端口

    centos7.5修改默认SSH端口 linux SSH默认端口是22,不修改的话存在一定的风险,要么是被人恶意扫描,要么会被人破解或者攻击,所以我们需要修改默认的SSH端口 1.修改22端口为620 ...

  10. 简单谈谈java中匿名内部类构造函数?

    先看看下面的代码能不能编译通过: public static void main(String[] args) {List l1 = new ArrayList();List l2 = new Arr ...