[BZOJ 2821] 作诗(Poetize) 【分块】
题目链接:BZOJ - 2821
题目分析
因为强制在线了,所以无法用莫队..可以使用分块来做。
做法是,将 n 个数分成 n/x 个块,每个块大小为 x 。先预处理出 f[i][j] ,表示从第 i 个块到第 j 个块的出现次数为偶数的数的个数。
这个复杂度是 n * (n / x) 的。
然后把数与位置存在结构体里,按照数字第一关键字,位置为第二关键字排序。这样是为了方便之后二分查找 [l, r] 中 Num 出现了几次。
对于每次询问,先把答案加上中间包含的整块的答案。然后对于两边至多 2x 个数,单独处理,二分求出它们在中间整块中出现的次数,以更新答案。
更新的思路大概是:看加上当前这个数后,这个数出现的次数是从odd -> even 还是从 even -> odd ,还是 0 -> 1,然后选择 ++Ans 或是 --Ans 或是什么也不做。
处理询问的复杂度是 n * x * logn 。
分析 x 的最优大小。总复杂度为 (n^2 / x) + (nlogn * x) ,由均值不等式得,当 n^2 / x == nlogn * x 时,总复杂度最小,所以 x 的最优值为 x = sqrt(n / logn) 。
代码
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm> using namespace std; const int MaxN = 100000 + 5, MaxBlk = 1300 + 5; inline void Read(int &num) {
char c = getchar();
while (c < '0' || c > '9') c = getchar();
num = c - '0'; c = getchar();
while (c >= '0' && c <= '9') {
num = num * 10 + c - '0';
c = getchar();
}
} int n, m, Cset, Ans, BlkSize, LastBlk;
int A[MaxN], First[MaxN], Last[MaxN], Cnt[MaxN], f[MaxBlk][MaxBlk], L[MaxBlk], R[MaxBlk]; struct ES
{
int Pos, Num;
bool operator < (const ES &b) const {
if (Num == b.Num) return Pos < b.Pos;
return Num < b.Num;
}
} E[MaxN]; int Get(int s, int t, int Num) {
if (s > t || s > E[Last[Num]].Pos || t < E[First[Num]].Pos) return 0;
int l, r, mid, pl, pr;
l = First[Num]; r = Last[Num];
while (l <= r) {
mid = (l + r) >> 1;
if (E[mid].Pos >= s) {
pl = mid;
r = mid - 1;
}
else l = mid + 1;
}
l = First[Num]; r = Last[Num];
while (l <= r) {
mid = (l + r) >> 1;
if (E[mid].Pos <= t) {
pr = mid;
l = mid + 1;
}
else r = mid - 1;
}
return pr - pl + 1;
} int main()
{
//Init
Read(n); Read(Cset); Read(m);
for (int i = 1; i <= n; ++i) Read(A[i]);
BlkSize = (int)sqrt((double)n / log((double)n) * log(2.0));
LastBlk = (n - 1) / BlkSize + 1;
for (int i = 1; i <= LastBlk; ++i) {
L[i] = (i - 1) * BlkSize + 1;
R[i] = i * BlkSize;
}
R[LastBlk] = n;
memset(Cnt, 0, sizeof(Cnt));
for (int i = 1; i <= LastBlk; ++i) {
for (int j = 1; j <= Cset; ++j) Cnt[j] = 0;
for (int j = i; j <= LastBlk; ++j) {
f[i][j] = f[i][j - 1];
for (int k = L[j]; k <= R[j]; ++k) {
++Cnt[A[k]];
if ((Cnt[A[k]] & 1) == 0) ++f[i][j];
else if (Cnt[A[k]] != 1) --f[i][j];
}
}
}
for (int i = 1; i <= n; ++i) {
E[i].Pos = i; E[i].Num = A[i];
}
sort(E + 1, E + n + 1);
for (int i = 1; i <= n; ++i) {
if (First[E[i].Num] == 0) First[E[i].Num] = i;
Last[E[i].Num] = i;
}
//The array Cnt[] will be used later.
memset(Cnt, 0, sizeof(Cnt)); //Solve queries
int l, r, x, y, Lx, Ry, G;
Ans = 0;
for (int Case = 1; Case <= m; ++Case) {
Read(l); Read(r);
l = (l + Ans) % n + 1; r = (r + Ans) % n + 1;
if (l > r) swap(l, r);
x = (l - 1) / BlkSize + 1;
if (l != L[x]) ++x;
y = (r - 1) / BlkSize + 1;
if (r != R[y]) --y;
if (x > y) {
Ans = 0;
for (int i = l; i <= r; ++i) {
++Cnt[A[i]];
if ((Cnt[A[i]] & 1) == 0) ++Ans;
else if (Cnt[A[i]] != 1) --Ans;
}
for (int i = l; i <= r; ++i) --Cnt[A[i]];
}
else {
Lx = L[x]; Ry = R[y];
Ans = f[x][y];
for (int i = l; i < Lx; ++i) {
++Cnt[A[i]];
G = Get(Lx, Ry, A[i]);
if (((Cnt[A[i]] + G) & 1) == 0) ++Ans;
else if (Cnt[A[i]] + G != 1) --Ans;
}
for (int i = r; i > Ry; --i) {
++Cnt[A[i]];
G = Get(Lx, Ry, A[i]);
if (((Cnt[A[i]] + G) & 1) == 0) ++Ans;
else if (Cnt[A[i]] + G != 1) --Ans;
}
for (int i = l; i < Lx; ++i) --Cnt[A[i]];
for (int i = r; i > Ry; --i) --Cnt[A[i]];
}
printf("%d\n", Ans);
}
return 0;
}
[BZOJ 2821] 作诗(Poetize) 【分块】的更多相关文章
- BZOJ 2821: 作诗(Poetize)( 分块 )
分块,分成N^0.5块.O(N^1.5)预处理出sm[i][j]表示前i块中j的出现次数, ans[i][j]表示第i~j块的答案. 然后就可以O(N^0.5)回答询问了.总复杂度O((N+Q)N^0 ...
- BZOJ 2821作诗(Poetize) 分块
Description 有一个长度为n的序列,序列每个元素的范围[1,c],有m个询问x y,表示区间[x,y]中出现正偶数次的数的种类数. Solution 大力分块解决问题. 把序列分块,f[i] ...
- 2821: 作诗(Poetize)
2821: 作诗(Poetize) Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 1078 Solved: 348[Submit][Status] ...
- 【BZOJ2821】作诗(Poetize) 分块
Description 神犇SJY虐完HEOI之后给傻×LYD出了一题:SHY是T国的公主,平时的一大爱好是作诗.由于时间紧迫,SHY作完诗之后还要虐OI,于是SHY找来一篇长度为N的文章,阅读M次, ...
- [BZOJ 2821] 作诗
Link: BZOJ 2821 传送门 Solution: 一道类似区间众数的经典分块 由于个数为偶数这样的条件不能支持快速合并 因此要先$O(n*sqrt(n))$预处理出$pre[i][j]$表示 ...
- bzoj 2821 作诗 分块
基本思路和蒲公英一样 还是预处理出每两个块间的答案 询问时暴力跑两边的贡献 #include<cstdio> #include<cstring> #include<ios ...
- BZOJ2821 作诗(Poetize) 分块
题意 算法 经验总结 代码 题意 不带修改,查询数列[1,n]中[l,r]内的出现正偶数次的数的个数, 数列中的数 <= 1e5, n <= 1e5, 强制在线 算法 查询的内容: 区 ...
- 【分块】BZOJ2821 作诗(Poetize)
2821: 作诗(Poetize) Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 3265 Solved: 951[Submit][Status][ ...
- BZOJ2821 作诗(Poetize) 【分块】
BZOJ2821 作诗(Poetize) Description 神犇SJY虐完HEOI之后给傻×LYD出了一题: SHY是T国的公主,平时的一大爱好是作诗. 由于时间紧迫,SHY作完诗之后还要虐OI ...
随机推荐
- jQuery 实现上下,左右滑动
前几天的任务:http://t.sina.com.cn/ 的下滑效果. 渐变移动出足够的空白 -> 淡出最后一个 ->渐变移动出足够的空白 我们要做的是向左移动效果.这个效果用时需添加一 ...
- Linux下Postfix的配置和使用
Postfix为何物,详见:http://zh.wikipedia.org/wiki/Postfix 0.关于Postfix postfix的产生是为了替代传统的sendmail.相较于sendmai ...
- [Angular 2] 9. Replace ng-modle with #ref & events
Let's say you want to write a simple data bing app. when you type in a text box, somewhere in the ap ...
- CFS: 虚拟运行时间
http://edsionte.com/techblog/archives/4331 nice和prio的关系如下: #define NICE_TO_PRIO(nice) (MAX_RT_PRIO+n ...
- Systemtap kernel.trace("*") events source code
http://blog.163.com/digoal@126/blog/static/16387704020131014562216/
- 【Linux学习笔记】用nc实现两台主机间的文件传输(不需要输密码)
通常,可以用scp完成两台主机间的文件传输任务,但在主机间未建立信任关系的情况下,scp每次都需要输入密码,用起来感觉不是很方便,之前这篇笔记介绍过不用输入密码执行脚本或传输文件的方法,但对于一些临时 ...
- iOS View的Frame和bounds之区别,setbounds使用(深入探究)
前言: 在ios开发中经常遇到两个词Frame和bounds,本文主要阐述Frame和bound的区别,尤其是bound很绕,较难理解. 一.首先,看一下公认的资料: 先看到下面的代码你肯定就明白了一 ...
- 简单回顾C++中的字符串
C++中有两种字符串形式,一种是C语言字符数组,一般可以使用 char*指针来操作它:另一种是C++中基于标准库的string类型,这算是更高层次的抽象数据类型. 主要讨论一下string类型,既然是 ...
- hihocoder 北大网络赛 E.
给一个1000个点的多边形(从某个点依次按照外形给出每个节点),这个多边形不一定是凸多边形 再给一个圆,问这个多边形与圆相交区域的周长 我们将这个问题分成两个部分,第一部分是求线段在圆内的长度,第二部 ...
- 导出你的GAC Assembly中的DLLS
方法1: CMD命令中,进入C:\windows\assembly,然后XCOPY GAC_MSIL c:\temp /E 这样就得到了dlls了,以命名空间来分类. 如果想将dlls从集合中分出来, ...