【bzoj2141】排队 分块+树状数组

题目描述

排排坐,吃果果,生果甜嗦嗦,大家笑呵呵。你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和。红星幼儿园的小朋友们排起了长长地队伍,准备吃果果。不过因为小朋友们的身高有所区别,排成的队伍高低错乱,极不美观。设第i个小朋友的身高为hi,我们定义一个序列的杂乱程度为:满足i<j且hi>hj的(i,j)数量。幼儿园阿姨每次会选出两个小朋友,交换他们的位置,请你帮忙计算出每次交换后,序列的杂乱程度。为方便幼儿园阿姨统计,在未进行任何交换操作时,你也应该输出该序列的杂乱程度。

输入

第一行为一个正整数n,表示小朋友的数量;第二行包含n个由空格分隔的正整数h1,h2,…,hn,依次表示初始队列中小朋友的身高;第三行为一个正整数m,表示交换操作的次数;以下m行每行包含两个正整数ai和bi,表示交换位置ai与位置bi的小朋友。

输出

输出文件共m行,第i行一个正整数表示交换操作i结束后,序列的杂乱程度。

样例输入

3
130 150 140
2
2 3
1 3

样例输出

1
0
3


题解

分块+树状数组

题目描述不清,这里已补好,所求即逆序对的个数。

求逆序对我们可以使用树状数组。但是树状数组是离线的,也就是说每次交换都必须重新扫一遍,这样肯定会T。

由于每次交换时,除了这两个数及它们之间的数以外都是不需要改动的,只要找出中间的即可。

我们有分块大法。

把所有元素分成√n 个块,对每个块建立一个树状数组,就可以得出两个数之间所有整块的不同范围内数的个数。

然后对于多出来的那些数,直接暴力扫一下即可。由于它们都不是整块,所以不会有超过√n 个数。

这里偷了点懒没有if else,把符合条件加加减减直接变成加减条件成立性,应该不难理解。

时间复杂度O(n*√n*logn)。

需要注意的是两个数在同一个块内的处理,以及x>y的特判。

还有,题目中可能会出现相同的数,因此不能看作非小即大,需要分开处理。

#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
struct data
{
	int h , p;
}a[20010];
int v[20010] , top , f[150][20010];
bool cmp1(data a , data b)
{
	return a.h < b.h;
}
bool cmp2(data a , data b)
{
	return a.p < b.p;
}
void update(int p , int x , int a)
{
	int i;
	for(i = x ; i <= top ; i += i & (-i))
		f[p][i] += a;
}
int query(int p , int x)
{
	int i , ans = 0;
	for(i = x ; i ; i -= i & (-i))
		ans += f[p][i];
	return ans;
}
int main()
{
	int n , m , i , j , si , ans = 0 , x , y;
	scanf("%d" , &n);
	si = (int)sqrt(n);
	for(i = 0 ; i < n ; i ++ )
		scanf("%d" , &a[i].h) , a[i].p = i;
	sort(a , a + n , cmp1);
	for(i = 0 ; i < n ; i ++ )
	{
		if(a[i].h != v[top]) v[++top] = a[i].h;
		a[i].h = top;
	}
	sort(a , a + n , cmp2);
	for(i = 0 ; i < n ; i ++ )
	{
		for(j = 0 ; j <= i / si ; j ++ ) ans -= query(j , a[i].h);
		ans += i;
		update(i / si , a[i].h , 1);
	}
	printf("%d\n" , ans);
	scanf("%d" , &m);
	while(m -- )
	{
		scanf("%d%d" , &x , &y);
		x -- ; y -- ;
		if(x > y) swap(x , y);
		if(x / si == y / si)
			for(i = x + 1 ; i < y ; i ++ )
				ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
		else
		{
			for(i = x / si + 1 ; i < y / si ; i ++ )
				ans += (si - query(i , a[x].h)) + query(i , a[y].h - 1) - query(i , a[x].h - 1) - (si - query(i , a[y].h));
			for(i = x + 1 ; i < (x / si + 1) * si ; i ++ )
				ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
			for(i = y / si * si ; i < y ; i ++ )
				ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
		}
		ans += (a[x].h < a[y].h) - (a[x].h > a[y].h);
		update(x / si , a[x].h , -1) , update(y / si , a[y].h , -1);
		swap(a[x].h , a[y].h);
		update(x / si , a[x].h , 1) , update(y / si , a[y].h , 1);
		printf("%d\n" , ans);
	}
	return 0;
}
posted @ 2017-03-18 09:59  GXZlegend  阅读(327)  评论(0编辑  收藏  举报