BZOJ2821:作诗——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=2821
问题描述
神犇SJY虐完HEOI之后给傻×LYD出了一题:
SHY是T国的公主,平时的一大爱好是作诗。
由于时间紧迫,SHY作完诗之后还要虐OI,于是SHY找来一篇长度为N的文章,阅读M次,每次只阅读其中连续的一段[l,r],从这一段中选出一些汉字构成诗。因为SHY喜欢对偶,所以SHY规定最后选出的每个汉字都必须在[l,r]里出现了正偶数次。而且SHY认为选出的汉字的种类数(两个一样的汉字称为同一种)越多越好(为了拿到更多的素材!)。于是SHY请LYD安排选法。
LYD这种傻×当然不会了,于是向你请教……
问题简述:N个数,M组询问,每次问[l,r]中有多少个数出现正偶数次。
输入格式
输入第一行三个整数n、c以及m。表示文章字数、汉字的种类数、要选择M次。
第二行有n个整数,每个数Ai在[1, c]间,代表一个编码为Ai的汉字。
接下来m行每行两个整数l和r,设上一个询问的答案为ans(第一个询问时ans=0),令L=(l+ans)mod n+1, R=(r+ans)mod n+1,若L>R,交换L和R,则本次询问为[L,R]。
输出格式
输出共m行,每行一个整数,第i个数表示SHY第i次能选出的汉字的最多种类数。
样例输入
5 3 5
1 2 2 3 1
0 4
1 2
2 2
2 3
3 5
样例输出
2
0
0
0
1
————————————————————————————————————
分块思想定了就好办了。
注意这题无良卡时间和空间(虽然很大程度和bzoj老爷机有关)
我们还是预处理两个数组:
1.sum[i][j]:i元素在前j块出现的次数。
2.ans[i][j]:i~j块的正偶数个数的个数。
显然预处理之后对于询问我们就有了如下算法:
1.跨度<=2个块长度:直接暴力。
2.跨度>2个块长度:显然区间一定跨过了至少一些/个连续的块,这些连续的块的正偶数个数的个数,先更新到cur(即最终答案中),然后枚举非整块区间内的数i,统计i在非整块区间内的个数t,如果:
1.连续的块内没有i:那么我们判断t的奇偶即可,如果是偶数,cur++。
2.连续的块内有i:
设连续的块内i的个数为c。
1.c偶数,t奇数:cur--;
2.c奇数,t奇数:cur++;
返回cur即可。
Q1:ans数组怎么处理?
A1:我们可以很轻松处理sum数组,然后用和上面的方法一样的思想求解ans即可。
大致如下:
1.ans[i][j]=a[i][j-1];
(1.1:清空数组,注意只清当前块的数,不然TLE没话说)
2.统计j块元素的出现个数;
3.如同上面的方法判断即可。
#include<cstdio>
#include<queue>
#include<cctype>
#include<cstring>
#include<cmath>
#include<vector>
#include<algorithm>
using namespace std;
const int N=;
const int SQRTN=;
const int INF=;
inline int read(){
int X=,w=;char ch=;
while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
while(isdigit(ch))X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
int n,m,lim,s,cnt,a[N],bl[SQRTN],br[SQRTN];
int sum[N][SQRTN],ans[SQRTN][SQRTN],t[N];
bool vis[N];
inline void intoblock(){
for(int i=;i<=n;i++){
if(i%s==){br[cnt]=i-;bl[++cnt]=i;}
}
br[cnt]=n;bl[cnt+]=n+;
return;
}
inline void init(){
for(int i=;i<=cnt;i++){
for(int j=;j<=lim;j++)sum[j][i]=sum[j][i-];
for(int j=bl[i];j<=br[i];j++){
sum[a[j]][i]++;
}
}
for(int i=;i<=cnt;i++){
for(int j=i;j<=cnt;j++){
ans[i][j]=ans[i][j-];
for(int k=bl[j];k<=br[j];k++)t[a[k]]=vis[a[k]]=;
for(int k=bl[j];k<=br[j];k++)t[a[k]]++,vis[a[k]]=;
for(int k=bl[j];k<=br[j];k++){
if(vis[a[k]]){
int c=sum[a[k]][j-]-sum[a[k]][i-];
if(!c){
if(t[a[k]]%==)ans[i][j]++;
}else{
if(c%==&&t[a[k]]%!=)ans[i][j]--;
if(c%!=&&t[a[k]]%!=)ans[i][j]++;
}
vis[a[k]]=;
}
}
}
}
return;
}
inline int query(int l,int r){
memset(vis,,sizeof(vis));
int cur=;
if(r-l+<=*s){
for(int i=l;i<=r;i++){
if(!vis[a[i]])vis[a[i]]=t[a[i]]=;
else t[a[i]]++;
}
for(int i=l;i<=r;i++){
if(vis[a[i]]){
if(t[a[i]]%==)cur++;
vis[a[i]]=;
}
}
return cur;
}
int L=(l-)/s+,R=(r-)/s+;
cur=ans[L+][R-];
for(int i=l;i<=br[L];i++){
if(!vis[a[i]])vis[a[i]]=t[a[i]]=;
else t[a[i]]++;
}
for(int i=bl[R];i<=r;i++){
if(!vis[a[i]])vis[a[i]]=t[a[i]]=;
else t[a[i]]++;
}
for(int i=l;i<=br[L];i++){
if(vis[a[i]]){
int c=sum[a[i]][R-]-sum[a[i]][L];
if(!c){
if(t[a[i]]%==)cur++;
}else{
if(c%==&&t[a[i]]%!=)cur--;
if(c%!=&&t[a[i]]%!=)cur++;
}
vis[a[i]]=;
}
}
for(int i=bl[R];i<=r;i++){
if(vis[a[i]]){
int c=sum[a[i]][R-]-sum[a[i]][L];
if(!c){
if(t[a[i]]%==)cur++;
}else{
if(c%==&&t[a[i]]%!=)cur--;
if(c%!=&&t[a[i]]%!=)cur++;
}
vis[a[i]]=;
}
}
return cur;
}
int main(){
n=read();lim=read();m=read();s=sqrt(n);
for(int i=;i<=n;i++)a[i]=read();
intoblock();
init();
int pre=;
for(int i=;i<=m;i++){
int l=(read()+pre)%n+,r=(read()+pre)%n+;
if(l>r)swap(l,r);
printf("%d\n",pre=query(l,r));
}
return ;
}
BZOJ2821:作诗——题解的更多相关文章
- BZOJ2821 作诗(Poetize) 【分块】
BZOJ2821 作诗(Poetize) Description 神犇SJY虐完HEOI之后给傻×LYD出了一题: SHY是T国的公主,平时的一大爱好是作诗. 由于时间紧迫,SHY作完诗之后还要虐OI ...
- 【分块】BZOJ2821 作诗(Poetize)
2821: 作诗(Poetize) Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 3265 Solved: 951[Submit][Status][ ...
- 【BZOJ2821】作诗 题解(分块+前缀和)
前言:世间还有这么卡常的题…… ------------------ 题目链接 题目大意:给定长度为$n$的序列${a_i}$.有$m$次询问,问$[l,r]$内出现正偶数次的数字有多少个. 这题跟蒲 ...
- BZOJ2821 作诗(Poetize) 主席树 bitset
原文链接https://www.lydsy.com/JudgeOnline/problem.php?id=2821 题目传送门 - BZOJ2821 题意 $n$ 个数,$m$ 组询问,每次问 $[l ...
- bzoj2821作诗
http://www.lydsy.com/JudgeOnline/problem.php?id=2821 分块 我们把数列分成$\sqrt{N}$块 记$f[i][j]$表示第i块到第j块的答案,这个 ...
- BZOJ2821 作诗(分块)
和区间众数几乎一模一样的套路. // luogu-judger-enable-o2 #include<iostream> #include<cstdio> #include&l ...
- [BZOJ2821]作诗(分块)
题意 N个数,M组询问,每次问[l,r]中有多少个数出现正偶数次对于100%的数据,1≤n,c,m≤105 题解 (传说lyd省选的时候看错题 把题看成这个了 从此又多了一道分块神题)把N个数 ...
- bzoj2821: 作诗(Poetize)
分块 分sqrt(n)块 F[i][j]表示块i到块j的答案 s[i][j]表示数字i在前j块内出现了几次 #include <iostream> #include <cstdio& ...
- BZOJ2821 作诗(Poetize) 分块
题意 算法 经验总结 代码 题意 不带修改,查询数列[1,n]中[l,r]内的出现正偶数次的数的个数, 数列中的数 <= 1e5, n <= 1e5, 强制在线 算法 查询的内容: 区 ...
随机推荐
- R语言使用过程中出现的问题--读取EXCEL文件
方法一: 按照R导论中的方法,使用RODBC包, library(RODBC) channel<-odbcConnectExcel("file.xlsx") da2<- ...
- Awesome Flask
Awesome Flask A curated list of awesome Flask resources and plugins Awesome Flask Framework Admin i ...
- spark 相关
Spark为什么会比mapreduce快? 1.Spark减少了中间过程的磁盘读写,数据很多时候不需要落地,从而提升了效率. 2.Spark基于内存的读写,减少了磁盘IO.node数据交互的通信时间. ...
- C# 委托 / 跨线程访问UI / 线程间操作无效: 从不是创建控件“Form1”的线程访问它
C# 委托 / 跨线程访问UI / 线程间操作无效: 从不是创建控件“Form1”的线程访问它 网上的代码都比较复杂,还是这个简单 见代码, 简易解决办法: 主窗体代码 using System; ...
- c# enum 解析
解析定义的枚举 public enum OrderPaymentStatus { /// <summary> /// 未支付 /// </summary> [Descripti ...
- ReadyAPI 教程和示例(二)
声明:如果你想转载,请标明本篇博客的链接,请多多尊重原创,谢谢! 本篇使用的 ReadyAPI版本是2.5.0 接上一篇: 4.修改SoapUI测试 本节将演示如何为测试用例添加测试步骤以及更改请求参 ...
- js 加密 crypto-js des加密
js 加密 crypto-js https://www.npmjs.com/package/crypto-js DES 举例: js 引入: <script src=&quo ...
- NMAP-高级用法
1.报文分段 2.偏移 –mtu后面的数字是8的倍数 3.源端口欺骗 4.指定报文长度 5.ttl 6.mac地址伪造 0代表随机伪造 7.正常输出 8.输出为xml 9.输出为grep 10.输出所 ...
- POJ 3714 Raid(计算几何の最近点对)
Description After successive failures in the battles against the Union, the Empire retreated to its ...
- js经典试题之数组与函数
js经典试题之数组与函数 1:列举js的全局函数? 答案:JavaScript 中包含以下 7 个全局函数escape( ).eval( ).isFinite( ).isNaN( ).parseFlo ...