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, 强制在线 算法 查询的内容: 区 ...
随机推荐
- dva webpack 利用require.context加载多个model
dva redux数据管理都在models,根据业务不同models可能会有几十甚至上百的 [模块.js], 每次在index.js使用 app.model(require('./models/exa ...
- redis集群搭建(伪集群)
1.准备工作 去官网下载好你想要安装的redis版本,下载链接 2.搭建步骤 输入命令yum install gcc-c++安装好gcc环境,将下载好的redis安装包上传到 /usr/local 解 ...
- 《Git学习指南》学习笔记(一)
第二章 入门 git的安装 在Linux下,git的安装很简单.以我的系统Deepin/Ubuntu为例,只需在终端敲入sudo apt-get install git即可.其他Linux发行版可尝试 ...
- IDEA搭载Tomcat使用JSTL连接Oracle数据库
1.在IDEA中,JSTL库添加到WEB-INF/lib下面可以直接在JSP页面上通过 <%@ taglib uri="http://java.sun.com/jsp/jstl/cor ...
- HADOOP docker(十):hdfs 结构体系
1.简介2.namenode和datanode3.The File System Namespace 文件系统命名空间4.Data Replication 数据复制5.Replica Placemen ...
- HADOOP docker(五):hadoop用户代理 Proxy user
1.hadoop用户代理简介2.配置3.实验 1.hadoop用户代理简介 hadoop用户代理功能的作用是让超级用户superuser模拟一个普通用户来执行任务.比如用户joe通过oozie提交一个 ...
- [leetcode-676-Implement Magic Dictionary]
Implement a magic directory with buildDict, and search methods. For the method buildDict, you'll be ...
- C中的除法,商和余数的大小、符号如何确定
对于C中的除法,商和余数的大小.符号是如何确定的呢?在C89中,只规定了如果两个数为正整数,那么余数的符号为正,并且商的值是接近真实值的最大整数.比如5 / 2,那么商就是2,余数就是1.但是,C89 ...
- Alpha 冲刺(1/10)
队名 火箭少男100 组长博客 林燊大哥 作业博客 Alpha 冲鸭! 成员冲刺阶段情况 林燊(组长) 过去两天完成了哪些任务 协调各成员之间的工作,对多个目标检测及文字识别模型进行评估.实验,选取较 ...
- String 和 CharSequence 关系与区别
String 继承于CharSequence,也就是说String也是CharSequence类型. CharSequence是一个接口,它只包括length(), charAt(int index) ...