浅谈分块:https://www.cnblogs.com/AKMer/p/10369816.html

题目传送门:https://lydsy.com/JudgeOnline/problem.php?id=2724

对于每次询问的答案,要么是中间整块的众数,要么是在两侧不完整的块出现过的数。

根据这个性质,我们可以\(O(n\sqrt{n})\)求出每个块的众数和\(sum[i][j]\),表示从第一块到第\(i\)块内\(j\)出现了多少次。

然后再用区间\(dp\)在\(O(n\sqrt{n})\)的复杂度内求出\(mx[i][j]\),表示第\(i\)整块到第\(j\)整块的众数是多少。

对于每次询问,出现在两侧不完整的块的数,我们可以暴力扫描两侧把它们出现的次数丢到一个桶里,在加上在整块里出现的次数。

然后直接找次数最多的那个数就行了。

注意相同比大小要比原大小而不是离散化之后的大小。

时间复杂度:\(O(n\sqrt{n}+m\sqrt{n})\)

空间复杂度:\(O(n\sqrt{n})\)

代码如下:

#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int maxn=4e4+5; int n,m,cnt,block,top,lstans;
int v[maxn],tmp[maxn],bel[maxn],tot[maxn];
int L[205],R[205],stk[405],mx[205][205],sum[205][maxn]; int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
} int query(int l,int r,int v) {return sum[r][v]-sum[l-1][v];} int main() {
n=read(),m=read(),block=sqrt(n);
for(int i=1;i<=n;i++) {
tmp[i]=v[i]=read();
bel[i]=(i-1)/block+1;
if(bel[i]!=bel[i-1])
R[bel[i-1]]=i-1,L[bel[i]]=i;
}
R[bel[n]]=n,sort(tmp+1,tmp+n+1);
cnt=unique(tmp+1,tmp+n+1)-tmp-1;
for(int i=1;i<=n;i++)
v[i]=lower_bound(tmp+1,tmp+cnt+1,v[i])-tmp;
for(int i=1;i<=bel[n];i++) {
memcpy(sum[i],sum[i-1],sizeof(sum[i]));
int num2=0;
for(int j=L[i];j<=R[i];j++)
sum[i][v[j]]++;
for(int j=L[i];j<=R[i];j++) {
int num1=query(i,i,v[j]);
if((num1>num2)||(num1==num2&&tmp[v[j]]<tmp[mx[i][i]]))
mx[i][i]=v[j],num2=num1;
}
}
for(int len=2;len<=bel[n];len++)
for(int i=1;i+len-1<=bel[n];i++) {
int j=i+len-1,res=mx[i][j-1],num2=query(i,j,res);
for(int k=L[j];k<=R[j];k++) {
int num1=query(i,j,v[k]);
if((num1>num2)||(num1==num2&&tmp[v[k]]<tmp[res]))
res=v[k],num2=num1;
}
mx[i][j]=res;
}
while(m--) {
int l=read(),r=read(),res=0,num2=0;
l=(l+lstans-1)%n+1,r=(r+lstans-1)%n+1;
if(r<l)swap(l,r);
if(bel[l]==bel[r]) {
for(int i=l;i<=r;i++)
if((++tot[v[i]])==1)stk[++top]=v[i];
}
else {
for(int i=l;i<=R[bel[l]];i++)
if((++tot[v[i]])==1) {
stk[++top]=v[i];
tot[v[i]]+=query(bel[l]+1,bel[r]-1,v[i]);
}
for(int i=L[bel[r]];i<=r;i++)
if((++tot[v[i]])==1) {
stk[++top]=v[i];
tot[v[i]]+=query(bel[l]+1,bel[r]-1,v[i]);
}
int tmp=mx[bel[l]+1][bel[r]-1];
if(!tot[tmp]) {
stk[++top]=tmp;
tot[tmp]=query(bel[l]+1,bel[r]-1,tmp);
}
}
for(int i=1;i<=top;i++) {
int num1=tot[stk[i]];
if((num1>num2)||(num1==num2&&tmp[stk[i]]<tmp[res]))
res=stk[i],num2=num1;
}
while(top)tot[stk[top--]]=0;
lstans=tmp[res];
printf("%d\n",lstans);
}
return 0;
}

BZOJ2724:[Violet 6]蒲公英的更多相关文章

  1. [BZOJ2724][Violet 6]蒲公英

    [BZOJ2724][Violet 6]蒲公英 试题描述 输入 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod n + 1 输出 输入示 ...

  2. BZOJ2724 [Violet 6]蒲公英 分块

    原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ2724.html 题目传送门 - BZOJ2724 题意 求区间最小众数,强制在线. $n$ 个数,$m ...

  3. bzoj2724: [Violet 6]蒲公英(离散化+分块)

    我好弱啊..这题调了2天QwQ 题目大意:给定一个长度为n(n<=40000)的序列,m(m<=50000)次询问l~r之间出现次数最多的数.(区间众数) 这题如果用主席树就可以不用处理一 ...

  4. bzoj2724: [Violet 6]蒲公英 分块 区间众数 论algorithm与vector的正确打开方式

    这个,要处理各个数的话得先离散,我用的桶. 我们先把每个块里的和每个块区间的众数找出来,那么在查询的时候,可能成为[l,r]区间的众数的数只有中间区间的众数和两边的数. 证明:若不是这里的数连区间的众 ...

  5. 【分块】bzoj2724 [Violet 6]蒲公英

    分块,离散化,预处理出: ①前i块中x出现的次数(差分): ②第i块到第j块中的众数是谁,出现了多少次. 询问的时候,对于整块的部分直接获得答案:对于零散的部分,暴力统计每个数出现的次数,加上差分的结 ...

  6. bzoj2724: [Violet 6]蒲公英(分块)

    传送门 md调了一个晚上最后发现竟然是空间开小了……明明算出来够的…… 讲真其实我以前不太瞧得起分块,觉得这种基于暴力的数据结构一点美感都没有.然而今天做了这道分块的题才发现分块的暴力之美(如果我空间 ...

  7. 【BZOJ2724】[Violet 6]蒲公英 分块+二分

    [BZOJ2724][Violet 6]蒲公英 Description Input 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod n ...

  8. BZOJ 2724: [Violet 6]蒲公英

    2724: [Violet 6]蒲公英 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 1633  Solved: 563[Submit][Status ...

  9. BZOJ 2724: [Violet 6]蒲公英( 分块 )

    虽然AC了但是时间惨不忍睹...不科学....怎么会那么慢呢... 无修改的区间众数..分块, 预处理出Mode[i][j]表示第i块到第j块的众数, sum[i][j]表示前i块j出现次数(前缀和, ...

  10. BZOJ_2724_[Violet 6]蒲公英_分块

    BZOJ_2724_[Violet 6]蒲公英_分块 Description Input 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod ...

随机推荐

  1. Shell编程之Expect自动化交互程序

    一.Expect自动化交互程序 1.spawn命令 通过spawn执行一个命令或程序,之后所有的Expect操作都会在这个执行过的命令或程序进程中进行,包括自动交互功能. 语法: spawn [ 选项 ...

  2. CentOS 6.5下Redmine的安装配置

    首先引用百度介绍下redmine: Redmine是用Ruby开发的基于web的项目管理软件,是用ROR框架开发的一套跨平台项目管理系统,据说是源于Basecamp的ror版而来,支持多种数据库,有不 ...

  3. PHP面向对象之对象和引用

    在PHP中对象类型和简单变量类型表现可以说是大相径庭,很多数据类型都要可以在写时进行复制,如当写代码$a=$b时,两个变量因为赋予相同的值而告终.所以需要注意的是,这种情况用在对象时就会完全不同了. ...

  4. tensorFlow 神经网络2

    learnrate 太大容易跑飞,设置激活函数 可以一定程度上增加learnrate,不跑飞 self.saver = tf.train.Saver() 和 self.init_variable = ...

  5. java异常中的finally(一)

    finally是保证语句能一定执行的.不管程序是否会报错,我们把程序一定要执行的代码放在finally中. 比如说流的关闭,不管在读写的过程中是否报错,一定要关闭流,可以把流的关闭操作放在finall ...

  6. java RC4加密解密

    package com.dgut.app.utils; import java.lang.Byte; import java.util.UUID; public class RC4 { public ...

  7. 使用PHP+Swoole实现的网页即时聊天工具:PHPWebIM

    使用PHP+Swoole实现的网页即时聊天工具 全异步非阻塞Server,可以同时支持数百万TCP连接在线 同时支持websocket+comet2种兼容协议,可用于所有种类的浏览器包括IE 拥有完整 ...

  8. java编程将命令行参数转化为整数并实现求和的功能。

  9. PromiseJs

    (function() { var define, requireModule, require, requirejs; (function() { var registry = {}, seen = ...

  10. BW ON HANA 业务模型关系与数据取数

    在接到业务需求之后,我认为重要的是理清楚自己该做什么.来实现业务.由于不了解业务,还是走了很多弯路.本可以不用这么做,还是这么做了.自然你最傻瓜的按照用户的方式去实现是没有问题的. 会使后面的人难以维 ...