分块:

先预处理,将原序列分成长度为len的许多块,计算从第i块到第j块的答案,(可以做到O(n*n/len))。

每次询问时,将询问的区间分成三部分,:左边,中间,右边,中间是尽量大的一个块区间,其答案已经计算得到,左右两边加起来最多有2*len个元素,暴力计算其对答案的影响。O(q*len*f(n)),f(n)是暴力加入一个元素的代价。

这道题f(n)是log(n)

总的复杂度:f(n) = O( n*n/len + q*len*log(n) ),

当len = n*(q*log(n))-1/2时取最小值.

 /**************************************************************
Problem: 2821
User: idy002
Language: C++
Result: Accepted
Time:24352 ms
Memory:13728 kb
****************************************************************/ #include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define maxn 100010
#define maxs 1510
#define isok(a) ((a)>0&&!((a)&1))
using namespace std; int n, c, m;
int a[maxn], f[maxs][maxs];
int lx[maxn], rx[maxn], mno[maxn], stot;
int cnt[maxn], cur_ans;
int vs[maxn], vt[maxn], vp[maxn];
bool met[maxn]; struct Pair {
int w, pos;
Pair(){}
Pair( int w, int pos ) : w(w), pos(pos) {}
};
Pair vod[maxn];
bool cmpwp( const Pair & a, const Pair & b ) {
return a.w<b.w || (a.w==b.w && a.pos<b.pos);
}
bool cmpp( const Pair & a, const Pair & b ) {
return a.pos<b.pos;
} void part() {
int len = (int)(double)sqrt((double)n/(log((double)n)/log()))+;
stot = n/len + (n%len!=);
for( int i=; i<=stot; i++ ) {
lx[i] = rx[i-]+;
rx[i] = rx[i-]+len;
if( i==stot ) rx[i] = n;
for( int j=lx[i]; j<=rx[i]; j++ )
mno[j] = i;
}
}
inline void update( int oldv, int newv ) {
bool od, nw;
od = isok(oldv);
nw = isok(newv);
if( od && !nw ) cur_ans--;
else if( !od && nw ) cur_ans++;
} void prep() {
for( int i=; i<=n; i++ )
vod[i] = Pair(a[i],i);
sort( vod+, vod++n, cmpwp );
for( int i=; i<=n; i++ ) {
if( vod[i].w!=vod[i-].w )
vs[vod[i].w] = vt[vod[i-].w] = i;
}
vt[vod[n].w] = n+;
for( int i=; i<=n; i++ ) vp[i] = vod[i].pos; for( int i=; i<=stot; i++ ) {
for( int j=i; j<=stot; j++ ) {
for( int k=lx[j]; k<=rx[j]; k++ ) {
update( cnt[a[k]], cnt[a[k]]+ );
cnt[a[k]]++;
}
f[i][j] = cur_ans;
}
cur_ans = ;
memset( cnt, , sizeof(cnt) );
}
} int qu_cnt( int w, int l, int r ) {
return upper_bound( vp+vs[w], vp+vt[w], r )
- lower_bound( vp+vs[w], vp+vt[w], l );
}
int query( int l, int r ) {
int ml=mno[l], mr=mno[r];
int rt;
if( ml==mr ) {
for( int j=l; j<=r; j++ ) {
update( cnt[a[j]], cnt[a[j]]+ );
cnt[a[j]]+=;
}
rt = cur_ans;
cur_ans = ;
for( int j=l; j<=r; j++ )
cnt[a[j]]--;
return rt;
}
if( mno[l]==mno[l-] ) ml++;
if( mno[r]==mno[r+] ) mr--;
cur_ans = f[ml][mr];
for( int j=l; j<lx[ml]; j++ ) {
if( met[a[j]] ) continue;
met[a[j]] = true;
int t1 = qu_cnt(a[j],l,r);
int t2 = qu_cnt(a[j],lx[ml],rx[mr]);
update( t2, t1 );
}
for( int j=r; j>rx[mr]; j-- ) {
if( met[a[j]] ) continue;
met[a[j]] = true;
int t1 = qu_cnt(a[j],l,r);
int t2 = qu_cnt(a[j],lx[ml],rx[mr]);
update( t2, t1 );
}
rt = cur_ans; cur_ans = ;
for( int j=l; j<lx[ml]; j++ ) met[a[j]]=false;
for( int j=r; j>rx[mr]; j-- ) met[a[j]]=false;
return rt;
} int main() {
scanf( "%d%d%d", &n, &c, &m );
for( int i=; i<=n; i++ ) scanf( "%d", a+i );
part();
prep();
for( int i=,x=,l,r; i<=m; i++ ) {
scanf( "%d%d", &l, &r );
l = (l+x)%n+;
r = (r+x)%n+;
if( l>r ) swap(l,r);
printf( "%d\n", x=query(l,r) );
}
}

bzoj 2821 分块的更多相关文章

  1. bzoj 2821 分块处理

    大题思路就是分块,将n个数分成sqrt(n)个块,然后 处理出一个w数组,w[i,j]代表第i个块到第j个块的答案 那么对于每组询问l,r如果l,r在同一个块中,直接暴力做就行了 如果不在同一个块中, ...

  2. BZOJ 2821 分块+二分

    题意: N个数,M组询问,每次问[l,r]中有多少个数出现正偶数次. 思路: 把N个数分成sqrt(n)块,预处理d[i][j]表示第i块起点到第j块末尾的答案 枚举起点i,并维护一个数组记录每个数到 ...

  3. [BZOJ 2821] 作诗(Poetize) 【分块】

    题目链接:BZOJ - 2821 题目分析 因为强制在线了,所以无法用莫队..可以使用分块来做. 做法是,将 n 个数分成 n/x 个块,每个块大小为 x .先预处理出 f[i][j] ,表示从第 i ...

  4. [BZOJ 2821] 作诗

    Link: BZOJ 2821 传送门 Solution: 一道类似区间众数的经典分块 由于个数为偶数这样的条件不能支持快速合并 因此要先$O(n*sqrt(n))$预处理出$pre[i][j]$表示 ...

  5. 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 ...

  6. BZOJ 2821作诗(Poetize) 分块

    Description 有一个长度为n的序列,序列每个元素的范围[1,c],有m个询问x y,表示区间[x,y]中出现正偶数次的数的种类数. Solution 大力分块解决问题. 把序列分块,f[i] ...

  7. bzoj 2821 作诗 分块

    基本思路和蒲公英一样 还是预处理出每两个块间的答案 询问时暴力跑两边的贡献 #include<cstdio> #include<cstring> #include<ios ...

  8. 【BZOJ 2821】作诗

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=2821 [算法] 如果不强制在线,显然莫队是可以解决此题的,那么,强制在线怎么办呢? ...

  9. bzoj 2741 分块+可持久化trie

    多个询问l,r,求所有子区间异或和中最大是多少 强制在线 做法: 分块+可持久化trie 1.对于每块的左端点i,预处理出i到任意一个j,()i,j)间所有子区间异或和中最大为多少,复杂度O(\(n\ ...

随机推荐

  1. 基于canvas实现的fontawesome动态图标

    由于还没有全部实现,实现了一些demo,demo地址在 https://github.com/jiangzhenfei/canvas-fontawesome 实现了动态loading 实现动态电池充电 ...

  2. 45、如何使用python删除一个文件?

    若想利用python删除windows里的文件,这里需要使用os模块!那接下来就看看利用os模块是如何删除文件的! 具体实现方法如下! os.remove(path) 删除文件 path. 如果pat ...

  3. Traffic-Server配置(待补充和更新)

    Server 5.3.2 测试1.裸盘:remap.configmap http://192.168.227.131 http://192.168.227.131:8080 #traffic_serv ...

  4. CSS实现箭头效果

    有时候网页中使用箭头以增强效果,一般的做法是使用图片,今天我们使用CSSCSS来实现“箭头效果”,使用CSS我们必须兼容所有浏览器(IE6.7.8.9.10.+),Chrome,Firefox,Ope ...

  5. nth-child,nth-last-child,only-child,nth-of-type,nth-last-of-type,only-of-type,first-of-type,last-of-type,first-child,last-child伪类区别和用法

    我将这坨伪类分成三组,第一组:nth-child,nth-last-child,only-child第二组:nth-of-type,nth-last-of-type,第三组:first-of-tpye ...

  6. python并发编程之multiprocessing进程(二)

    python的multiprocessing模块是用来创建多进程的,下面对multiprocessing总结一下使用记录. 系列文章 python并发编程之threading线程(一) python并 ...

  7. 牛奶ddw如何通过以太坊钱包实现互相打赏

    很多朋友不清楚如何转账ddw,但是万能的网友是无敌的,这两天就自己摸索的一点经验总结下今天的转账经验. 1. 提取到自己的账户 这个大家都知道如何操作,使用官方的钱包 在“日日盈app”中点击&quo ...

  8. C++ 和Java继承机制的比较

    摘要: C++支持类的多继承,而Java采用类的单继承.C++中的继承成分只有类(模板属于带参数的类,结构和联合是特殊的类),Java中除了类还有接口的继承,而且允许接口的多继承,可以间接地实现类多继 ...

  9. 记一次java内存溢出的解决过程

    注:本文主要记录这次解决内存溢出问题的过程而不是具体问题. 最近在写一个搜索引擎,使用倒排索引结构进行文档检索,保存索引的基本思想是先将倒排列表保存到内存中一个有序Map里(TreeMap),然后当内 ...

  10. 编辑器之Sublime Text3、Notepad++

    Sublime text 3 破解版是一款极其强大的代码编辑器,又是一款可以代替记事本的文本编辑器.Sublime text 3拥有着美观的界面和实用的功能,既能够完成代码的编辑又能够完成文本编辑,还 ...