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, 强制在线 算法 查询的内容: 区 ...
随机推荐
- 离线安装Sharepoint工具
1. 首先安装操作系统,Windows Server 2008 R2,可以是企业版,也可以是数据中心版.然后再安装上SP1. 2. 在"服务管理"里面,添加角色,安装IIS. ...
- python import vs from import
https://stackoverflow.com/questions/9439480/from-import-vs-import
- 现实世界中的 Python
Python 有多稳定? 非常稳定. 自 1991 年起大约每隔 6 到 18 个月就会推出新的稳定发布版,这种状态看来还将持续下去. 目前主要发布版本的间隔通常为 18 个月左右. 开发者也会推出旧 ...
- 《Git学习指南》学习笔记(三)
多次提交 提交一般分未两步:add和commit. add将修改存入到索引(index)或叫暂存区(staging area)中. status命令 status命令会出现三种可能的状态: chang ...
- python程序设计——面向对象程序设计:类
理解面向对象 基本原则是,计算机程序由多个能够起到子程序作用的单元或对象组合而成 关键性观念是,数据以及对数据的操作封装在一起,组成一个相互依存.不可分割的整体,即对象 python面向对象特性 完全 ...
- 【python 3.6】从网站抓图并存放到本地路径
#!/usr/bin/python # -*- coding: UTF-8 -*- _author_ = 'BH8ANK' import urllib.request import re import ...
- 【Paper】Deep & Cross Network for Ad Click Predictions
目录 背景 相关工作 主要贡献 核心思想 Embedding和Stacking层 交叉网络(Cross Network) 深度网络(Deep Network) 组合层(Combination Laye ...
- Saruman's Army(贪心)
Saruman the White must lead his army along a straight path from Isengard to Helm’s Deep. To keep tra ...
- Pipeline组测试说明
PIPELINE组测试报告 前言:我们组与学霸系统的其他两个小组共同合作开发,组成学霸系统的团体工作.作为学霸系统的一环,我们组起到承上启下的作用,因此,面向群体以及功能实现都是为给下一个组的工作做好 ...
- 【android】实现手指滑动来切换activity(转)
http://code.eoe.cn/115 1.jpg外部引用 原始文档 MainActivity.java外部引用 原始文档 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 ...