题目传送门

题意

长度为n的序列,有m次询问,每次询问求\([l,r]\) 间的众数,如果有多个,输出最小的那个

\(n\le 4\times 10^4,m\le 5\times 10^5,a_i\le10^9\)

分析

题目中要求在线(询问用上次答案加密)。众数不具有“区间可加性”,所以需要分块。

假设分成\(T\)块,每块长度\(L=N/T\)。每次询问\([l,r]\),设\(l\)处在\(p\)块,\(r\)处在\(q\)块,则区间分为三部分。

  • 开头的\([l,L)\)
  • \(第p+1\sim q-1块构成的区间[L,R]\)
  • 结尾的\((R,r]\)

    显然序列在整个询问区间的众数只可能出现在\([L,R]\)中的众数,或者头尾出现的数字中。

    可以预处理所有分块区间的众数\(d[l][r]\)表示\(l\)块到\(r\)块中的众数。复杂度\(O(TN)\)

    对于头尾的部分,暴力枚举每个数字,通过预先存在每个数字出现的位置,这样就可以二分查询该数字\([l,r]\)出现的次数,然后选取最大的即可。复杂度\(NM/T*log(N)\)

    所以总复杂度为\(O(TN+NM/T*log(N))\),又N和M几乎同级,可得\(T = \sqrt{Mlog(N)}\)时较优。

    更多细节请看代码

思路&步骤

  1. 存下每个数字,离散化,处理每个数字出现的位置,储存在STL::vector里面
  2. 分块,预处理整区间的众数
  3. 查询,先取整个区间的答案,然后两头扫
#include <bits/stdc++.h>
using namespace std;
const int N = 40010;
const int M = 500010;
/*
be[i] 表示 i 在第几块
id[i] 表示a[i]在离散化数组中的位置
blc为块大小
cnt[i] 预处理时需要的数组,表示 i 的个数
d[i][j]表示 i块与 j块之间的众数
v[i] 表示 数字 i 出现的位置
g为离散化vector
*/
int be[N],id[N],a[N],n,m;
int blc,cnt[N],d[1200][1200];
vector<int> v[N],g; int getID(int x){
return lower_bound(g.begin(),g.end(),x) - g.begin();
}
//预处理整块区间众数
void build(int x){
int res = -1;
for(int i=0;i<g.size();i++)cnt[i] = 0;
for(int i=(x-1)*blc+1;i<=n;i++){
cnt[id[i]] ++;
if(res == -1 || cnt[id[i]] > cnt[res] || (cnt[id[i]] == cnt[res] && id[i] < res)){
res = id[i];
}
d[x][be[i]] = res;
}
}
//询问[l,r]间x的个数
int ask(int l,int r,int x){
return upper_bound(v[x].begin(),v[x].end(),r) - lower_bound(v[x].begin(),v[x].end(),l);
}
int query(int l,int r){
//mxcnt为当前众数的个数,res为答案
int mxcnt = 0,res = -1;
if(be[r] - be[l] > 1){
res = d[be[l]+1][be[r]-1];
mxcnt = ask(l,r,res);
}
if(be[r] == be[l]){
for(int i=l;i<=r;i++){
int cnt = ask(l,r,id[i]);
if(res == -1 || mxcnt < cnt || (mxcnt == cnt && id[i] < res)){
res = id[i];
mxcnt = cnt;
}
}
}
else{
for(int i=l;i<=be[l] * blc;i++){
int cnt = ask(l,r,id[i]);
if(res == -1 || mxcnt < cnt || (mxcnt == cnt && id[i] < res)){
res = id[i];
mxcnt = cnt;
}
}
for(int i=(be[r]-1) * blc;i<=r;i++){
int cnt = ask(l,r,id[i]);
if(res == -1 || mxcnt < cnt || (mxcnt == cnt && id[i] < res)){
res = id[i];
mxcnt = cnt;
}
}
}
return res;
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++){
scanf("%d",&a[i]);
g.push_back(a[i]);
}
sort(g.begin(),g.end());g.erase(unique(g.begin(),g.end()),g.end());
for(int i=1;i<=n;i++){
id[i] = getID(a[i]);
v[id[i]].push_back(i);
}
blc = max(1,(int)(n/sqrt(m * log2(n))));
for(int i=1;i<=n;i++){
be[i] = (i-1) / blc + 1;
}
for(int i=1;i<=be[n];i++)
build(i);
int res = 0;
while(m--){
int l,r;
scanf("%d%d",&l,&r);
l = (l + res - 1) % n + 1;
r = (r + res - 1) % n + 1;
if(l > r)swap(l,r);
res = g[query(l,r)];
printf("%d\n",res);
}
return 0;
}

Luogu4168 蒲公英 (分块)的更多相关文章

  1. luogu4168蒲公英(区间众数)

    luogu4168蒲公英(区间众数) 给定n个数,m个区间询问,问每个询问中的众数是什么. 题面很漂亮,大家可以去看一下. 对于区间众数,由于区间的答案不能由子区间简单的找出来,所以似乎不能用树形结构 ...

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

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

  3. [Violet]蒲公英 分块

    发现写算法专题老是写不动,,,, 所以就先把我在luogu上的题解搬过来吧! 题目大意:查询区间众数,无修改,强制在线 乍一看是一道恐怖的题,仔细一看发现并没有那么难: 大致思路是这样的,首先我们要充 ...

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

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

  5. luogu P4168 蒲公英+ 分块学习笔记

    传送门 题目描述 在乡下的小路旁种着许多蒲公英,而我们的问题正是与这些蒲公英有关. 为了简化起见,我们把所有的蒲公英看成一个长度为n的序列\((a_1,a_2..a_n)\),其中 \(a_i\)为一 ...

  6. 洛谷P4168 蒲公英 分块处理区间众数模板

    题面. 许久以前我还不怎么去机房的时候,一位大佬好像一直在做这道题,他称这道题目为"大分块". 其实这道题目的思想不只可以用于处理区间众数,还可以处理很多区间数值相关问题. 让我们 ...

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

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

  8. bzoj 2724 蒲公英 分块

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

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

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

随机推荐

  1. 8. 格式化器大一统 -- Spring的Formatter抽象

    目录 ✍前言 本文提纲 版本约定 ✍正文 Printer&Parser Formatter 时间日期格式化 Date类型 代码示例 JSR 310类型 整合DateTimeFormatter ...

  2. day121:MoFang:植物的状态改动(幼苗→成长期)&植物的浇水功能

    目录 1.当果树种植以后在celery的异步任务中调整浇水的状态 2.客户端通过倒计时判断时间,显示浇水道具 3.客户端判断当前种植物状态控制图标的显示和隐藏 4.当用户单击浇水图标, 则根据当前果树 ...

  3. php利用腾讯ip分享计划获取地理位置示例分享

    <?php function getIPLoc_QQ($queryIP){ $url = 'http://ip.qq.com/cgi-bin/searchip?searchip1='.$quer ...

  4. 【剑指 Offer】11.旋转数组的最小数字

    题目描述 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素.例如,数组 [3,4,5,1,2] 为 [1,2,3,4,5] 的 ...

  5. LeetCode662 二叉树最大宽度

    给定一个二叉树,编写一个函数来获取这个树的最大宽度.树的宽度是所有层中的最大宽度.这个二叉树与满二叉树(full binary tree)结构相同,但一些节点为空. 每一层的宽度被定义为两个端点(该层 ...

  6. Kubernetes K8S之kube-prometheus概述与部署

    Kubernetes K8S之kube-prometheus概述与部署 主机配置规划 服务器名称(hostname) 系统版本 配置 内网IP 外网IP(模拟) k8s-master CentOS7. ...

  7. P1220 关路灯(区间规划)

    题目描述 某一村庄在一条路线上安装了n盏路灯,每盏灯的功率有大有小(即同一段时间内消耗的电量有多有少).老张就住在这条路中间某一路灯旁,他有一项工作就是每天早上天亮时一盏一盏地关掉这些路灯. 为了给村 ...

  8. Xctf攻防世界—crypto—Normal_RSA

    下载压缩包后打开,看到两个文件flag.enc和pubkey.pem,根据文件名我们知道应该是密文及公钥 这里我们使用一款工具进行解密 工具链接:https://github.com/3summer/ ...

  9. SparkStreaming和Kafka基于Direct Approach如何管理offset实现exactly once

    在之前的文章<解析SparkStreaming和Kafka集成的两种方式>中已详细介绍SparkStreaming和Kafka集成主要有Receiver based Approach和Di ...

  10. springmvc 字符串转日期格式

    http://www.mamicode.com/info-detail-2485490.html