归并排序求逆序对
2015-11-21 07:49
330 查看
归并排序求逆序对
我们知道归并排序是递归的对左右两边区间排序,然后将已经有序的左边区间和右边区间合并。合并的时候,假设左边区间范围是 l <= i < mid 右边的范围是 mid <= j < r ,如果a[i] <= a[j],那么没有逆序对产生, 如果 a[i] > a[j] ,那么 所有 a[i] .. a[i+i] ... a[mid-1]这些数都是大于a[j]的,因为左边区间已经有序。所以只要每次遇到 a[i] > a[j], ans += mid - i,就能求出所有的逆序对。
题目:http://poj.org/problem?id=1804
源代码
#include <iostream>#include <cstdio>
using namespace std;
const int maxn = 1111;
int a[maxn],tmp[maxn];
int ans;
void Merge(int l,int m, int r){
int i=l;
int j = m;
int k = l;
while(i < m && j < r){
if(a[i] > a[j]){
tmp[k++] = a[j++];
ans += m - i ;
}
else
tmp[k++] = a[i++];
}
while(i < m) tmp[k++] = a[i++];
while(j < r) tmp[k++] = a[j++];
for(int i=l;i<r;i++)
a[i] = tmp[i];
}
void Merge_Sort(int l,int r){
if(r - l > 1){
int m = (l + r) >>1;
Merge_Sort(l,m);
Merge_Sort(m,r);
Merge(l,m,r);
}
}
int main(){
int t;
scanf("%d",&t);
int cnt = 1;
while(t--){
int n;
scanf("%d",&n);
for(int i=0;i<n;i++)
scanf("%d",&a[i]);
ans = 0;
Merge_Sort(0,n);
// for(int i=0;i<n;i++)
// printf("%d\n",a[i]);
printf("Scenario #%d:\n%d\n\n",cnt++,ans);
}
return 0;
}
相关文章推荐
- 在命令行用 sort 进行排序
- 文件遍历排序函数
- C#选择排序法实例分析
- C#插入法排序算法实例分析
- Lua教程(七):数据结构详解
- C#实现Datatable排序的方法
- 解析从源码分析常见的基于Array的数据结构动态扩容机制的详解
- SQLSERVER的排序问题结果不是想要的
- Windows Powershell排序和分组管道结果
- C#数据结构揭秘一
- C#通过IComparable实现ListT.sort()排序
- C#选择法排序实例分析
- SQL学习笔记四 聚合函数、排序方法
- C#对list列表进行随机排序的方法
- 一根网线内的8根线哪4根是传输数据的,哪四根是防干扰的
- 数据结构之Treap详解
- C#折半插入排序算法实现方法
- SQL进行排序、分组、统计的10个新技巧分享
- C++实现位图排序实例
- 基于C++实现的各种内部排序算法汇总