bzoj2821: 作诗(Poetize)

原创 2015年08月12日 10:05:20

传送门:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2821

思路:分块大法好。。。

分成sqrt(n)块,先预处理出连续的块的答案,f[i][j]表示第i块到第j块的答案。

然后再开一个前缀和数组sum[i][j]表示前i个块第j种字符出现的次数。

对于一组询问[l,r],先得出连续的块的答案。

对于分散的两端的块,暴力考虑每个数对答案的贡献即可。

#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
const int maxn=100010,maxb=320;
using namespace std;
int n,c,m,sum[maxb][maxn],f[maxb][maxb],a[maxn],sz,cnt,bel[maxn],l[maxn],r[maxn],tsum[maxn],ans,tmp[maxn];

void work(){
	int x,y,L,R;scanf("%d%d",&x,&y);
	x=(x+ans)%n+1,y=(y+ans)%n+1,ans=0;
	if (x>y) swap(x,y);
	//printf("%d %d\n",x,y);
	L=bel[x],R=bel[y];
	if (L==R){
		for (int i=x;i<=y;i++) tsum[a[i]]=0;
		for (int i=x;i<=y;i++)
			if ((++tsum[a[i]])!=1)
				ans+=(tsum[a[i]]&1)?-1:1;
	}
	else{
		ans=f[L+1][R-1];
		//printf("ans%d %d %d\n",L,R,ans);
		for (int i=x;i<=r[L];i++) tsum[a[i]]=sum[R-1][a[i]]-sum[L][a[i]];
		for (int i=l[R];i<=y;i++) tsum[a[i]]=sum[R-1][a[i]]-sum[L][a[i]];
		//for (int i=1;i<=c;i++) printf("tsum%d\n",tsum[i]);
		for (int i=x;i<=r[L];i++)
			if (++tsum[a[i]]!=1)
				ans+=(tsum[a[i]]&1)?-1:1;
		for (int i=l[R];i<=y;i++)
			if (++tsum[a[i]]!=1)
				ans+=(tsum[a[i]]&1)?-1:1;
	}
	printf("%d\n",ans);
}

int main(){
	scanf("%d%d%d",&n,&c,&m),sz=sqrt(n),cnt=n/sz+(n%sz!=0);
	for (int i=1;i<=n;i++) scanf("%d",&a[i]);
	for (int i=1;i<=n;i++) bel[i]=(i-1)/sz+1;
	for (int i=1;i<=n;i++){r[bel[i]]=i;if (!l[bel[i]]) l[bel[i]]=i;}
	for (int i=1;i<=cnt;i++){
		for (int j=1;j<=c;j++) sum[i][j]=sum[i-1][j];
		for (int j=l[i];j<=r[i];j++) sum[i][a[j]]++;
	}
	for (int i=1;i<=cnt;i++){
		for (int j=1;j<=c;j++) tsum[j]=0;int tmp=0;
		for (int j=i;j<=cnt;j++){
			for (int k=l[j];k<=r[j];k++)
				if (tsum[a[k]]++) tmp+=(tsum[a[k]]&1)?-1:1;
			f[i][j]=tmp;
		}
	}
	//for (int i=1;i<=cnt;i++) printf("%d %d\n",l[i],r[i]);
	/*for (int i=1;i<=cnt;i++,puts(""))
		for (int j=1;j<=cnt;j++)
			printf("%d ",f[i][j]);*/
	for (int i=1;i<=m;i++) work();
	return 0;
}


版权声明:本文为博主原创文章,未经博主允许不得转载。

【bzoj 2821】 作诗 题意&题解&代码(C++)

分块
  • DERITt
  • DERITt
  • 2016年03月17日 21:14
  • 338

BZOJ 2821 作诗(Poetize) 分块

题目大意:给定一个序列 多次求区间中多少个数出现次数为偶数次 强制在线 很神的一道分块的题……记得刚进BZ坑的时候看到这道题50秒特别惊奇0.0 然后我就作死去交了个死循环0.0 看了很多题解 都...
  • PoPoQQQ
  • PoPoQQQ
  • 2014年10月22日 10:14
  • 1995

bzoj2821: 作诗(Poetize)

传送门:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2821 思路:分块大法好。。。 分成sqrt(n)块,先预处理出连续的块的答案,f[i...
  • thy_asdf
  • thy_asdf
  • 2015年08月12日 10:05
  • 582

[bzoj2821]作诗

题目大意询问区间出现次数为正偶数的数的个数分块大法好经典分块方法。 预处理ans[i,j]表示第i块到第j块的答案,sum[i,j]表示前j块中元素i出现的次数。 然后搞一波就行了。 bzoj上...
  • WerKeyTom_FTD
  • WerKeyTom_FTD
  • 2016年03月07日 14:03
  • 446

bzoj2821 作诗(Poetize)

神奇的分块题,求大家帮助
  • AaronGZK
  • AaronGZK
  • 2016年03月03日 23:01
  • 1165

【bzoj2821】作诗(Poetize)

*题目描述: 神犇SJY虐完HEOI之后给傻×LYD出了一题:SHY是T国的公主,平时的一大爱好是作诗。由于时间紧迫,SHY作完诗 之后还要虐OI,于是SHY找来一篇长度为N的文章,阅读M次,每次...
  • qq_33442848
  • qq_33442848
  • 2016年05月23日 11:33
  • 123

BZOJ2821: 作诗(Poetize)(分块)

传送门 题意 给你一个区间,每次查询(l,r),问(l,r)中数字出现偶数次的种类数。 题解 分块大法好。。。 分析: 1.查询(l,r)中的数字出现偶数次的种类数,可以等价于完整块中数字种类数及...
  • qq_35649707
  • qq_35649707
  • 2017年07月15日 17:22
  • 91

BZOJ2821: 作诗(Poetize)

分块,因为没有修改的操作,在线询问,所以可以预处理出s[i][j]表示1到i块j出现的次数,f[i][j]表示i到j块有多少数出现次数为偶数,询问时两端的块暴力做一下就行了 题目卡空间(有毒),把块...
  • L_0_Forever_LF
  • L_0_Forever_LF
  • 2016年10月28日 09:52
  • 555

bzoj2821[作诗(Poetize)]

BZOJ2821 作诗
  • qq_37321281
  • qq_37321281
  • 2017年07月19日 21:19
  • 84

分块-bzoj2821: 作诗(Poetize)

http://hzwer.com/3663.html 我的程序好垃圾,常数好像很大的样子,本来是WA,然后改好之后交上去超时; 改来改去半天还是TLE,后来把分块长度变成sqrt(n/log2(n...
  • largecub233
  • largecub233
  • 2017年02月22日 18:28
  • 167
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:bzoj2821: 作诗(Poetize)
举报原因:
原因补充:

(最多只允许输入30个字)