【题目链接】

https://www.lydsy.com/JudgeOnline/problem.php?id=2724

【算法】

分块算法在线维护区间众数

分块算法的精髓就在于 : 大段维护,局部朴素

这一题,我们可以将序列分成T段,那么每一段的长度就是(N/T)

对于每个询问,设l处于第p段,r处于第q段,那么 :

若p = q,用朴素算法计算出区间众数即可

否则,将这个序列分为三段 :

1.[L,R[p]] 2. [L[p+1],R[q-1]] 3. [L[q],r]

显然,区间众数只可能是 [L[p+1],R[q-1]]中的众数或[L,R[p]]中的一个数,或[L[q],r]中的一个数

不妨预处理所有以“段边界”为端点每个数出现的次数和区间众数

    当T取sqrt3(N)(开三次方)时,时间复杂度是非常优秀的 : O(n^(5/3))

【代码】

#include<bits/stdc++.h>
using namespace std;
#define MAXN 40010
const int INF = 2e9; int i,j,k,n,m,l,r,l0,r0,lastans,t,block,len;
int a[MAXN],val[MAXN],L[],R[],pos[MAXN],rk[MAXN];
int cnt[][][MAXN],mx[][],d[][]; inline int query(int l,int r)
{
int i;
static int sum[MAXN];
int mx = ,ret = ;
int p = pos[l],q = pos[r];
if (p == q)
{
for (i = l; i <= r; i++)
{
sum[rk[i]]++;
if (sum[rk[i]] > mx || (sum[rk[i]] == mx && rk[i] < ret))
{
mx = sum[rk[i]];
ret = rk[i];
}
}
for (i = l; i <= r; i++) sum[rk[i]]--;
return ret;
} else
{
ret = d[p+][q-];
mx = cnt[p+][q-][ret];
for (i = l; i <= R[p]; i++)
{
cnt[p+][q-][rk[i]]++;
if (cnt[p+][q-][rk[i]] > mx || (cnt[p+][q-][rk[i]] == mx && rk[i] < ret))
{
mx = cnt[p+][q-][rk[i]];
ret = rk[i];
}
}
for (i = L[q]; i <= r; i++)
{
cnt[p+][q-][rk[i]]++;
if (cnt[p+][q-][rk[i]] > mx || (cnt[p+][q-][rk[i]] == mx && rk[i] < ret))
{
mx = cnt[p+][q-][rk[i]];
ret = rk[i];
}
}
for (i = l; i <= R[p]; i++) cnt[p+][q-][rk[i]]--;
for (i = L[q]; i <= r; i++) cnt[p+][q-][rk[i]]--;
return ret;
}
} int main()
{ scanf("%d%d",&n,&m);
for (i = ; i <= n; i++)
{
scanf("%d",&a[i]);
val[++len] = a[i];
}
sort(val+,val+len+);
len = unique(val+,val+len+) - val - ;
for (i = ; i <= n; i++) rk[i] = lower_bound(val+,val+len+,a[i]) - val;
block = (int)pow(n*1.0,1.0/);
if (block) t = n / block;
for (i = ; i <= block; i++)
{
L[i] = (i - ) * t + ;
R[i] = i * t;
}
if (R[block] < n)
{
block++;
L[block] = R[block-] + ;
R[block] = n;
}
for (i = ; i <= block; i++)
{
for (j = L[i]; j <= R[i]; j++)
{
pos[j] = i;
}
}
for (i = ; i <= block; i++)
{
for (j = i; j <= block; j++)
{
for (k = L[i]; k <= R[j]; k++)
cnt[i][j][rk[k]]++;
for (k = ; k <= len; k++)
{
if (cnt[i][j][k] > mx[i][j] || (cnt[i][j][k] == mx[i][j] && k < d[i][j]))
{
mx[i][j] = cnt[i][j][k];
d[i][j] = k;
}
}
}
}
lastans = ;
for (i = ; i <= m; i++)
{
scanf("%d%d",&l0,&r0);
l = (l0 + lastans - ) % n + ;
r = (r0 + lastans - ) % n + ;
if (l > r) swap(l,r);
printf("%d\n",lastans = val[query(l,r)]);
} return ; }

【BZOJ 2724】 蒲公英的更多相关文章

  1. bzoj 2724 蒲公英 分块

    分块,预处理出每两个块范围内的众数,然后在暴力枚举块外的进行比较 那么怎么知道每一个数出现的次数呢?离散后,对于每一个数,维护一个动态数组就好了 #include<cstdio> #inc ...

  2. BZOJ 2724 蒲公英 | 分块模板题

    题意 给出一个序列,在线询问区间众数.如果众数有多个,输出最小的那个. 题解 这是一道分块模板题. 一个询问的区间的众数,可能是中间"整块"区间的众数,也可能是左右两侧零散的数中的 ...

  3. BZOJ 2724蒲公英 (分块) 【内有块大小证明】

    题面 luogu传送门 分析 先分块,设块大小为x(之后我们会证明块大小取何值会更优) 步骤1 把所有的数离散化,然后对每个值开一个vector pos[i],pos[i]存储数i出现的位置 我们设查 ...

  4. [BZOJ 2724] [Violet 6] 蒲公英 【分块】

    题目链接:BZOJ - 2724 题目分析 这道题和 BZOJ-2821 作诗 那道题几乎是一样的,就是直接分块,每块大小 sqrt(n) ,然后将数字按照数值为第一关键字,位置为第二关键字排序,方便 ...

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

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

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

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

  7. 【BZOJ 2724】 2724: [Violet 6]蒲公英 (区间众数不带修改版本)

    2724: [Violet 6]蒲公英 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 1908  Solved: 678 Description In ...

  8. 【刷题】BZOJ 2724 [Violet 6]蒲公英

    Description Input 修正一下 l = (l_0 + x - 1) mod n + 1, r = (r_0 + x - 1) mod n + 1 Output Sample Input ...

  9. BZOJ 2724: [Violet 6]蒲公英 [分块 区间众数]

    传送门 题面太美不忍不放 分块分块 这种题的一个特点是只有查询,通常需要预处理:加入修改的话需要暴力重构预处理 预处理$f[i][j]$为第i块到第j块的众数,显然$f[i][j]=max{f[i][ ...

随机推荐

  1. Windows环境下使用强大的wget工具

    安装 下载[http://www.interlog.com/~tcharron/wgetwin.html] 解压到目录 比如我解压到D:\Tool\wget 添加wget环境变量,这样使用就更方便了, ...

  2. C#服务控件UpdatePanel的局部刷新与属性AutoPostBack回传

    C#服务控件许多都有AutoPostBack这一属性(AutoPostBack意思是自动回传,也就是说此控件值更改后是否和服务器进行交互),如下代码所示: protected void Textbox ...

  3. lua迭代

    迭代 function enum(array) local index = 1 return function() --返回迭代函数 local ret = array[index] index = ...

  4. java是值传递,还是引用传递?

    原文地址:http://blog.csdn.net/zxmzfbdc/article/details/5401960  java到底是值传递,还是引用传递?以前国内的java开发者有过很多争论,由于& ...

  5. GEF中连接的实现

    在GEF绘图笔想象中复杂许多,需要很多组件的依赖与支持,稍微弄错一个引用,或一个操作调试起来就比较麻烦,下面列一下实现一个连接线功能所需要实现的类及添加的方法 建议大图查看. 相关代码:参考<G ...

  6. RFID 知识的学习

    * 部分资料来自我们博士的PPT,部分来自网络和他人的论文. * 我们使用的教材是清华大学出版社出版的<智能卡技术(第四版)——IC卡.RFID标签与物联网(清华大学计算机系列教材)>(王 ...

  7. SpringMVC(五)@RequestHeader和@CookieValue

    通过使用@RequestHeader获取请求头 通过使用@CookieValue获取cookie值 代码: 1: @Controller 2: public class TestHeader_Cook ...

  8. .apply和.call用法和区别

    apply:方法能劫持另外一个对象的方法,继承另外一个对象的属性. Function.apply(obj,args)方法能接收两个参数obj:这个对象将代替Function类里this对象args:这 ...

  9. 初级模拟电路:1-2 PN结与二极管

    回到目录 1.   掺杂半导体 上面我们分析了本征半导体的导电情况,但由于本征半导体的导电能力很低,没什么太大用处.所以,一般我们会对本征半导体材料进行掺杂,即使只添加了千分之一的杂质,也足以改变半导 ...

  10. Linux常用命令学习随笔

    1.ls命令 就是list的缩写,通过ls 命令不仅可以查看linux文件夹包含的文件,而且可以查看文件权限(包括目录.文件夹.文件权限)查看目录信息等等 常用参数搭配: ls -a 列出目录所有文 ...