标签:树状 com namespace 输出 size nbsp int sample 变化
原文地址:http://www.cnblogs.com/GXZlegend/p/6805224.html
题目描述
输入
第一行一个正整数n,表示Mato的资料份数。
第二行由空格隔开的n个正整数,第i个表示编号为i的资料的大小。
第三行一个正整数q,表示Mato会看几天资料。
之后q行每行两个正整数l、r,表示Mato这天看[l,r]区间的文件。
输出
q行,每行一个正整数,表示Mato这天需要交换的次数。
样例输入
4
1 4 2 3
2
1 2
2 4
样例输出
0
2
题解
离散化+莫队算法+树状数组
首先有交换次数等于逆序对数
然后问题就转化为如何求一段区间的逆序对数。
由于[l,r]可推出[l-1,r]或[l,r+1],可以考虑莫队算法。
先将询问排序,然后每次加入或删除元素时统计一下有多少逆序对变化即可,其中细节较多。
注意题中没给资料大小的范围,所以需要先离散化。
#include <cstdio>
#include <cmath>
#include <algorithm>
#define N 50010
using namespace std;
struct DATA
{
int num , pos;
}a[N];
struct QUERY
{
int l , r , bl , id;
}q[N];
int st[N] , top , val[N] , f[N] , ans[N];
bool cmp1(DATA a , DATA b)
{
return a.num < b.num;
}
bool cmp2(QUERY a , QUERY b)
{
return a.bl == b.bl ? a.r < b.r : a.bl < b.bl;
}
void update(int x , int a)
{
int i;
for(i = x ; i <= top ; i += i & -i) f[i] += a;
}
int query(int x)
{
int i , ans = 0;
for(i = x ; i ; i -= i & -i) ans += f[i];
return ans;
}
int main()
{
int n , m , si , i , lp = 1 , rp = 0 , now = 0;
scanf("%d" , &n) , si = (int)sqrt(n);
for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &a[i].num) , a[i].pos = i;
sort(a + 1 , a + n + 1 , cmp1);
for(i = 1 ; i <= n ; i ++ )
{
if(a[i].num != st[top]) st[++top] = a[i].num;
val[a[i].pos] = top;
}
scanf("%d" , &m);
for(i = 1 ; i <= m ; i ++ ) scanf("%d%d" , &q[i].l , &q[i].r) , q[i].bl = (q[i].l - 1) / si , q[i].id = i;
sort(q + 1 , q + m + 1 , cmp2);
for(i = 1 ; i <= m ; i ++ )
{
while(lp < q[i].l) now -= query(val[lp] - 1) , update(val[lp] , -1) , lp ++ ;
while(lp > q[i].l) lp -- , now += query(val[lp] - 1) , update(val[lp] , 1);
while(rp > q[i].r) now -= rp - lp + 1 - query(val[rp]) , update(val[rp] , -1) , rp -- ;
while(rp < q[i].r) rp ++ , now += rp - lp - query(val[rp]) , update(val[rp] , 1);
ans[q[i].id] = now;
}
for(i = 1 ; i <= m ; i ++ ) printf("%d\n" , ans[i]);
return 0;
}
【bzoj3289】Mato的文件管理 离散化+莫队算法+树状数组
标签:树状 com namespace 输出 size nbsp int sample 变化
原文地址:http://www.cnblogs.com/GXZlegend/p/6805224.html