标签:树状 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