题目链接

时间限制:10000ms
单点时限:1000ms
内存限制:256MB

描述

给定一个1-N的排列A1, A2, ... AN,如果Ai和Aj满足i < j且Ai > Aj,我们就称(Ai, Aj)是一个逆序对。

求A1, A2 ... AN中所有逆序对的数目。

输入

第一行包含一个整数N。

第二行包含N个两两不同整数A1, A2, ... AN。(1 <= Ai <= N)

对于60%的数据 1 <= N <= 1000

对于100%的数据 1 <= N <= 100000

输出

一个整数代表答案

样例输入
5
3 2 4 5 1
样例输出
5
 求逆序对,两种方法,一个是用树形数组做,一个是用归并排序做。两种都试了一遍,有一个地方比较坑就是数组类型必须是long long。不然会WA,题目却没有明显信息说要long long。。这个地方可以说是很坏坏了。
 
树形数组ac代码:
 #include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<iomanip>
using namespace std; //1524 : 逆序对
const int maxn = 1e5 + ;
int n;
long long a[maxn], b[maxn];
int lowbit(int i)
{
return i&-i;
}
void update(int i)
{
while (i <= n)
{
b[i]++;
i += lowbit(i);
}
}
long long sum(int i)
{
long long cnt = ;
while (i)
{
cnt += b[i];
i -= lowbit(i);
}
return cnt;
}
int main()
{
cin >> n;
for (int i = ;i <= n;i++)
{
cin >> a[i];
}
memset(b, , sizeof(b));
long long res = ;
for (int i = n;i >= ;i--)
{
update(a[i]);
res += sum(a[i] - );
}
cout << res << endl;
return ;
}

树形数组这个东西还是比较有趣的。这个算法可以说是比较天才了。

具体的树形数组详解可以看这个博主的http://blog.csdn.net/ljd4305/article/details/10101535

归并排序的ac代码:

 #include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<iomanip>
using namespace std; //1524 : 逆序对
const int maxn = 1e5 + ;
int n;
long long a[maxn],b[maxn];
long long cnt; void merge_sort(long long * a, int l, int r)
{
int mid = (l + r) >> ;
int m = l, n = mid + ;
int pos = l;
while (m<=mid&&n<=r)
{
if (a[m] < a[n])
{
b[pos++] = a[m++];
}
else
{
b[pos++] = a[n++];
cnt+=mid-m+;
}
} while (m <= mid) b[pos++] = a[m++];
while (n <= r) b[pos++] = a[n++];
for (int i = l;i <= r;i++)
a[i] = b[i];
}
void merge(long long * a, int l, int r)
{
if (l >= r) return;
int mid = (l + r) >>;
merge(a, l, mid);
merge(a, mid+, r);
merge_sort(a, l, r);
} int main()
{
cnt = ;
cin >> n;
for (int i = ;i <= n;i++)
{
cin >> a[i];
}
merge(a, , n);
cout << cnt << endl;
return ;
}

归并排序是将数列a[l,h]分成两半a[l,mid]和a[mid+1,h]分别进行归并排序,然后再将这两半合并起来。在合并的过程中(设l<=i<=mid,mid+1<=j<=h),当a[i]<=a[j]时,并不产生逆序数;当a[i]>a[j]时,在前半部分中比a[i]大的数都比a[j]大,将a[j]放在a[i]前面的话,逆序数要加上mid+1-i。因此,可以在归并排序中的合并过程中计算逆序数。

可以参考http://blog.csdn.net/acdreamers/article/details/16849761

hihocoder1524的更多相关文章

  1. 树形数组 java

    2^k求法 int lowbit(int x) { return x&(-x); } lowbit()的返回值就是 2^k 次方的值. 基本树形数组的模板 import java.util.* ...

随机推荐

  1. EF分别使用IQueryable和IEnumerable实现更新和删除

    缺点 使用IQueryable无法跟踪,无法监控sql,无法使用SaveChanges(). 优点 使用IQueryable简单粗暴. class Program { static void Main ...

  2. [转] 浅谈session,cookie,sessionStorage,localStorage的区别及应用场景

    浏览器的缓存机制提供了可以将用户数据存储在客户端上的方式,可以利用cookie,session等跟服务端进行数据交互. 一.cookie和session cookie和session都是用来跟踪浏览器 ...

  3. 【BZOJ2698】染色

    题解: 首先比较显然的是查询每个点被覆盖的概率,算完之后概率m次方 既然是计数题 考虑容斥 我们会发现这样是求n长度的区间能存多少种 我们考虑直接递推 从n到n+1 多的方案数一定要覆盖n+1,所以就 ...

  4. [WC2014]紫荆花之恋

    题解: 首先考虑点分治 dis(i,u)+dis(i,v)<=value[u]+value[v] 移项就很容易发现用平衡树可以很简单的维护这个东西 但是有重复,需要在下一层的每个平衡树内减去这个 ...

  5. 3998: [TJOI2015]弦论

    题解: 每个点的size值就是这个从根-它出现的次数 如果相同只算一次就全部赋值为1就可以了 代码: #include <bits/stdc++.h> #define ll long lo ...

  6. 【BZOJ】3022: [Balkan2012]The Best Teams

    原题链接 题面 (为啥这题没有题面-- 给出\(N\)个人,和年龄\(age_{i},skill_{i}\) 然后给出\(M\)个询问,就是年龄在\(a\)以下选不超过\(k\)个人 要求选择的人水平 ...

  7. k8s 相关命令

    kompose convert -f docker-compose-pro.yml k8s数据卷挂载: https://blog.csdn.net/wlhdo71920145/article/deta ...

  8. java的IO流的一些测试

    public class Demo02 { public static void main(String[] args) { //获取当前毫秒值,用于测试 long l = System.curren ...

  9. web发展阶段简介

     web1.0.web2.0和web3.0的区别前言: 其实并没有什么所谓的2.0.3.0,因为你没法准确的界定它是什么样的应用,也没法界定它是什么时候开始的,什么时候结束,它只是互联网本身发展的一种 ...

  10. C++中全排列算法函数next_permutation的使用方法

    首先,先看对next_permutation函数的解释: http://www.cplusplus.com/reference/algorithm/next_permutation/?kw=next_ ...