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:作诗——题解的更多相关文章

  1. BZOJ2821 作诗(Poetize) 【分块】

    BZOJ2821 作诗(Poetize) Description 神犇SJY虐完HEOI之后给傻×LYD出了一题: SHY是T国的公主,平时的一大爱好是作诗. 由于时间紧迫,SHY作完诗之后还要虐OI ...

  2. 【分块】BZOJ2821 作诗(Poetize)

    2821: 作诗(Poetize) Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 3265  Solved: 951[Submit][Status][ ...

  3. 【BZOJ2821】作诗 题解(分块+前缀和)

    前言:世间还有这么卡常的题…… ------------------ 题目链接 题目大意:给定长度为$n$的序列${a_i}$.有$m$次询问,问$[l,r]$内出现正偶数次的数字有多少个. 这题跟蒲 ...

  4. BZOJ2821 作诗(Poetize) 主席树 bitset

    原文链接https://www.lydsy.com/JudgeOnline/problem.php?id=2821 题目传送门 - BZOJ2821 题意 $n$ 个数,$m$ 组询问,每次问 $[l ...

  5. bzoj2821作诗

    http://www.lydsy.com/JudgeOnline/problem.php?id=2821 分块 我们把数列分成$\sqrt{N}$块 记$f[i][j]$表示第i块到第j块的答案,这个 ...

  6. BZOJ2821 作诗(分块)

    和区间众数几乎一模一样的套路. // luogu-judger-enable-o2 #include<iostream> #include<cstdio> #include&l ...

  7. [BZOJ2821]作诗(分块)

    题意 N个数,M组询问,每次问[l,r]中有多少个数出现正偶数次对于100%的数据,1≤n,c,m≤105 题解 (传说lyd省选的时候看错题   把题看成这个了   从此又多了一道分块神题)把N个数 ...

  8. bzoj2821: 作诗(Poetize)

    分块 分sqrt(n)块 F[i][j]表示块i到块j的答案 s[i][j]表示数字i在前j块内出现了几次 #include <iostream> #include <cstdio& ...

  9. BZOJ2821 作诗(Poetize) 分块

    题意 算法 经验总结 代码 题意 不带修改,查询数列[1,n]中[l,r]内的出现正偶数次的数的个数, 数列中的数 <= 1e5, n <= 1e5, 强制在线 算法 ​ 查询的内容: 区 ...

随机推荐

  1. 离线安装Sharepoint工具

    1. 首先安装操作系统,Windows Server 2008 R2,可以是企业版,也可以是数据中心版.然后再安装上SP1. 2. 在"服务管理"里面,添加角色,安装IIS.    ...

  2. python import vs from import

    https://stackoverflow.com/questions/9439480/from-import-vs-import

  3. 现实世界中的 Python

    Python 有多稳定? 非常稳定. 自 1991 年起大约每隔 6 到 18 个月就会推出新的稳定发布版,这种状态看来还将持续下去. 目前主要发布版本的间隔通常为 18 个月左右. 开发者也会推出旧 ...

  4. 《Git学习指南》学习笔记(三)

    多次提交 提交一般分未两步:add和commit. add将修改存入到索引(index)或叫暂存区(staging area)中. status命令 status命令会出现三种可能的状态: chang ...

  5. python程序设计——面向对象程序设计:类

    理解面向对象 基本原则是,计算机程序由多个能够起到子程序作用的单元或对象组合而成 关键性观念是,数据以及对数据的操作封装在一起,组成一个相互依存.不可分割的整体,即对象 python面向对象特性 完全 ...

  6. 【python 3.6】从网站抓图并存放到本地路径

    #!/usr/bin/python # -*- coding: UTF-8 -*- _author_ = 'BH8ANK' import urllib.request import re import ...

  7. 【Paper】Deep & Cross Network for Ad Click Predictions

    目录 背景 相关工作 主要贡献 核心思想 Embedding和Stacking层 交叉网络(Cross Network) 深度网络(Deep Network) 组合层(Combination Laye ...

  8. Saruman's Army(贪心)

    Saruman the White must lead his army along a straight path from Isengard to Helm’s Deep. To keep tra ...

  9. Pipeline组测试说明

    PIPELINE组测试报告 前言:我们组与学霸系统的其他两个小组共同合作开发,组成学霸系统的团体工作.作为学霸系统的一环,我们组起到承上启下的作用,因此,面向群体以及功能实现都是为给下一个组的工作做好 ...

  10. 【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 ...