#根号分治,前缀和,双指针#CF1446D2 Frequency Problem (Hard Version)
题目
给定一个长度为 \(n\) 的序列,问是否存在一个最长的区间使得至少存在两个众数。
分析
实际上 Easy Version 是用来启发大于根号的做法的。
众数可以说有一个性质吧,答案区间中的其中一个众数一定是整个序列的众数。
当然如果整个序列有多个众数答案就是 \(n\),如果只有一种数答案就是 \(0\)。
只需要考虑一种众数的情况,先看 Easy Version(数字种类不超过一百)。
如果数字种类数足够少,直接枚举数字种类,那么相当于一段区间众数和该种数字出现次数相等。
那么将众数视为 \(-1\),该种数字视为 \(1\),即求是否存在一段区间和为零,那直接前缀和记录最早位置即可。
但是 Hard Version 的时候不能所有数字都算一遍,
考虑枚举出现次数 \(T\),然后用一个双指针求出对于每一个右端点 \(r\),求出最小的 \(l\) 使得 \([l,r]\) 数字出现次数不超过 \(T\)。
这样只要有至少两个数字出现次数为 \(T\),\([l,r]\) 就是一段符合要求的区间。
将两种方法结合一下,\(T\) 只枚举到根号,枚举数字种类只枚举出现次数超过根号的数字,这样就是 \(O(n\sqrt{n})\) 的
代码
#include <cstdio>
#include <cctype>
using namespace std;
const int N=200011,bl=400;
int n,mx,se,cnt[N],a[N],c[N],las[N<<1],ans;
int iut(){
int ans=0; char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=ans*10+c-48,c=getchar();
return ans;
}
int max(int a,int b){return a>b?a:b;}
int main(){
n=iut();
for (int i=1;i<=n;++i) ++cnt[a[i]=iut()];
for (int i=1;i<=n;++i)
if (cnt[i]>cnt[mx]) se=mx,mx=i;
else if (cnt[i]>cnt[se]) se=i;
if (!se) return !printf("0");
if (cnt[mx]==cnt[se]) return !printf("%d",n);
for (int T=1;T<=bl;++T){
int now=0;
for (int i=1,j=1;i<=n;++i){
for (;c[a[i]]==T;--c[a[j++]])
if (c[a[j]]==T) --now;
if (++c[a[i]]==T) ++now;
if (now>1) ans=max(ans,i-j+1);
}
for (int i=0;i<=n;++i) c[i]=0;
}
for (int i=1;i<=n;++i)
if (cnt[i]>bl&&i!=mx){
for (int j=0;j<=2*n;++j) las[j]=-1; las[n]=0;
for (int j=1,s=0;j<=n;++j){
if (a[j]==i) ++s;
else if (a[j]==mx) --s;
if (las[s+n]>=0) ans=max(ans,j-las[s+n]);
else las[s+n]=j;
}
}
return !printf("%d",ans);
}
#根号分治,前缀和,双指针#CF1446D2 Frequency Problem (Hard Version)的更多相关文章
- 题解 CF1446D2 【Frequency Problem (Hard Version)】
给出一个跑得快一点的做法,洛谷最优解 (时间是第二名的 \(\frac{1}{2}\)), CF 第一页 D1 首先找到整个序列的众数 \(G\), 很容易证明答案序列中的两个众数中其中一个是 \(G ...
- Codeforces 1446D2 - Frequency Problem (Hard Version)(根分)
Codeforces 题面传送门 & 洛谷题面传送门 人菜结论题做不动/kk 首先考虑此题一个非常关键的结论:我们设整个数列的众数为 \(G\),那么在最优子段中,\(G\) 一定是该子段的众 ...
- CF587F-Duff is Mad【AC自动机,根号分治】
正题 题目链接:https://www.luogu.com.cn/problem/CF587F 题目大意 给出\(n\)个字符串\(s\).\(q\)次询问给出\(l,r,k\)要求输出\(s_{l. ...
- [CF587F]Duff is Mad[AC自动机+根号分治+分块]
题意 给你 \(n\) 个串 \(s_{1\cdots n}\) ,每次询问给出 \(l,r,k\) ,问在 \(s_{l\cdots r}\) 中出现了多少次 \(s_k\) . \(n,q,\su ...
- CF1039D-You Are Given a Tree【根号分治,贪心】
正题 题目链接:https://www.luogu.com.cn/problem/CF1039D 题目大意 给出\(n\)个点的一棵树,然后对于\(k\in[1,n]\)求每次使用一条长度为\(k\) ...
- UOJ#33-[UR #2]树上GCD【长链剖分,根号分治】
正题 题目链接:https://uoj.ac/problem/33 题目大意 给出\(n\)个点的一棵树 定义\(f(x,y)=gcd(\ dis(x,lca),dis(y,lca)\ )\). 对于 ...
- NOI.AC#2266-Bacteria【根号分治,倍增】
正题 题目链接:http://noi.ac/problem/2266 题目大意 给出\(n\)个点的一棵树,有一些边上有中转站(边长度为\(2\),中间有一个中转站),否则就是边长为\(1\). \( ...
- Codeforces 1039D You Are Given a Tree [根号分治,整体二分,贪心]
洛谷 Codeforces 根号分治真是妙啊. 思路 考虑对于单独的一个\(k\)如何计算答案. 与"赛道修建"非常相似,但那题要求边,这题要求点,所以更加简单. 在每一个点贪心地 ...
- BZOJ.4320.[ShangHai2006]Homework(根号分治 分块)
BZOJ \(\mathbb{mod}\)一个数\(y\)的最小值,可以考虑枚举剩余系,也就是枚举区间\([0,y),[y,2y),[2y,3y)...\)中的最小值(求后缀最小值也一样)更新答案,复 ...
- CF1039E Summer Oenothera Exhibition 贪心、根号分治、倍增、ST表
传送门 感谢这一篇博客的指导(Orzwxh) $PS$:默认数组下标为$1$到$N$ 首先很明显的贪心:每一次都选择尽可能长的区间 不妨设$d_i$表示在取当前$K$的情况下,左端点为$i$的所有满足 ...
随机推荐
- 将JavaBean对象转换为Map集合
使用jackson-databind可以将JavaBean对象属性转换为Map集合. 添加配置依赖: <dependency> <groupId>com.fasterxml.j ...
- 如何快速使用LayUI MINI框架
是什么 LayUI MINI是基于LayUI框架开发的一套最简洁.易用的后台框架模板,它已经是一个非常完整的脚手架,有现成的页面模板可以参考甚至是直接使用. 通常来说,如果我们准备开发一套管理系统的W ...
- 【手写信息搜集工具】ThunderSearch 闪电搜索器
ThunderSearch 闪电搜索器 项目地址:github Windows打包版 利用ZoomEye的官方api,结合开发文档,做了这么一个GUI界面的搜索器.目前支持查询host_search ...
- select_for_update悲观锁
例子,银行存款和撤销方法 1.用户A提取帐户 - 余额为100 $. 2.用户B提取帐户 - 余额为100 $. 3.用户B退出30 $ - 余额更新为100 $ - 30 $ = 70 $. 4.用 ...
- 【转载】nltk英文自定义分词
NLTK项目地址: https://github.com/nltk/nltk_data/tree/gh-pages/packages NLTK基础分词用例: https://www.cnblogs.c ...
- mysql-查询库中所有表名称或者某一张表的所有字段名称
-- 查询某一库中所有表的名称, SELECT a.TABLE_SCHEMA ,a.TABLE_NAME ,a.TABLE_COMMENT FROM information_schema.TABLES ...
- BIM+物联网,打开数字孪生世界之门
建筑行业一直在寻求创新和提高效率的方法,以满足日益复杂和迫切的建筑需求.近年来,数字孪生和物联网等新兴技术的崛起为建筑信息模型(BIM)应用带来了全新的可能性.数字孪生技术通过将实体建筑与其虚拟模型连 ...
- 使用C#和MemoryCache组件实现轮流调用APIKey以提高并发能力
文章信息 标题:使用C#和MemoryCache组件实现轮流调用API Key以提高并发能力的技巧 摘要:本文介绍了如何利用C#语言中的MemoryCache组件,结合并发编程技巧,实现轮流调用多个A ...
- Dyno File Utils - VSCode Extension 新建目录 新建文件 很好用
Dyno File Utils - VSCode Extension 新建目录 新建文件 很好用 快捷键 绑定了 ctrl + n
- K8s中Role(ClusterRole)资源类型rules字段详解
在Kubernetes(K8s)中,Role资源类型的rules字段用于定义哪些操作(verbs)可以在哪些资源(resources)上执行.Role是一种命名空间级别的资源,它允许你对命名空间内的资 ...