BZOJ 4527: K-D-Sequence
4527: K-D-Sequence
Time Limit: 20 Sec Memory Limit: 256 MB
Submit: 163 Solved: 66
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
4 3 2 8 6 2
Sample Output
//第一个测试样例的答案为包括数字 2,8,6 的子串——在加入数字 4 并且
排序之后,它变成了数列 2,4,6,8——公差为 2 的等差数列。
HINT
Source
分析:
考虑如果一个数列可以成为等差数列,其必要条件是任意$a[i]$%$d$都是相等的,也就是说,我们要找出连续的最长一段同余子串,使得其满足可以成为等差数列...
接的我们发现,可以把公差为$d$转化为公差为$1$,也就是满足$max(\frac{a[i]}{d})-min(\frac{a[i]}{d})+1-(j-i+1)<=k$,感觉还是很好理解的...
所以我们考虑枚举子串的右端点,然后找到最靠左的满足要求的点,发现我们可以用线段树来维护信息,维护三个值,$Min$代表的是最小的$min+l$,$Max$代表的是最小的$max+l$,$Sum$代表的是最小的$min+max+l$,然后我们每把右端点右移一位,就把当前的点影响的区间修改...
代码:
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<map>
//by NeighThorn
#define inf 2e9+7
using namespace std; const int maxn=200000+5; int n,k,d,top,ans,ansl,ansr,a[maxn],b[maxn],c[maxn],pre[maxn],stk[maxn],lmax[maxn],lmin[maxn]; map<int,int> mp; struct M{ int l,r;
long long Min,Max,Sum,tagMax,tagMin; inline void init(void){
Min=Max=Sum=tagMin=tagMax=inf;
} }tree[maxn<<2]; inline void pushdown(int tr,int l,int r,int mid){
int val;
if(tree[tr].tagMax!=inf)
val=tree[tr].tagMax,
tree[tr<<1].Max=val+l,
tree[tr<<1].tagMax=val,
tree[tr<<1].Sum=val+tree[tr<<1].Min,
tree[tr<<1|1].Max=val+mid+1,
tree[tr<<1|1].tagMax=val,
tree[tr<<1|1].Sum=val+tree[tr<<1|1].Min,
tree[tr].tagMax=inf;
if(tree[tr].tagMin!=inf)
val=tree[tr].tagMin,
tree[tr<<1].tagMin=val,
tree[tr<<1].Min=-val+l,
tree[tr<<1].Sum=-val+tree[tr<<1].Max,
tree[tr<<1|1].tagMin=val,
tree[tr<<1|1].Min=-val+mid+1,
tree[tr<<1|1].Sum=-val+tree[tr<<1|1].Max,
tree[tr].tagMin=inf;
} inline void update(int tr){
tree[tr].Min=min(tree[tr<<1].Min,tree[tr<<1|1].Min);
tree[tr].Max=min(tree[tr<<1].Max,tree[tr<<1|1].Max);
tree[tr].Sum=min(tree[tr<<1].Sum,tree[tr<<1|1].Sum);
} inline void build(int l,int r,int tr){
tree[tr].l=l,tree[tr].r=r,tree[tr].init();
if(l==r){
tree[tr].Max=c[l]+l,tree[tr].Min=-c[l]+l,tree[tr].Sum=l;
return;
}
int mid=(l+r)>>1;
build(l,mid,tr<<1);build(mid+1,r,tr<<1|1);
update(tr);
} inline void change(int l,int r,int id,int val,int tr){
if(tree[tr].l==l&&tree[tr].r==r){
if(id==0) tree[tr].tagMax=val,tree[tr].Max=val+tree[tr].l,tree[tr].Sum=tree[tr].Min+val;
if(id==1) tree[tr].tagMin=val,tree[tr].Min=-val+tree[tr].l,tree[tr].Sum=tree[tr].Max-val;
return;
}
int mid=(tree[tr].l+tree[tr].r)>>1;
pushdown(tr,tree[tr].l,tree[tr].r,mid);
if(r<=mid)
change(l,r,id,val,tr<<1);
else if(l>mid)
change(l,r,id,val,tr<<1|1);
else
change(l,mid,id,val,tr<<1),change(mid+1,r,id,val,tr<<1|1);
update(tr);
} inline int find(int tr,int val){
if(tree[tr].Sum>val) return n+1;
if(tree[tr].l==tree[tr].r) return tree[tr].l;
pushdown(tr,tree[tr].l,tree[tr].r,tree[tr].l+tree[tr].r>>1);
if(val>=tree[tr<<1].Sum) return find(tr<<1,val);
return find(tr<<1|1,val);
} inline int query(int l,int r,int tr,int val){
if(tree[tr].l==l&&tree[tr].r==r) return find(tr,val);
int mid=(tree[tr].l+tree[tr].r)>>1;
pushdown(tr,tree[tr].l,tree[tr].r,mid);
if(r<=mid) return query(l,r,tr<<1,val);
else if(l>mid) return query(l,r,tr<<1|1,val);
else{
int lala=query(l,mid,tr<<1,val);
if(lala==n+1)
return query(mid+1,r,tr<<1|1,val);
return lala;
}
} inline void solve(void){
for(int i=1,j;i<=n;i=j+1){
for(j=i;j<n&&a[j]==a[j+1];j++);
if(j-i+1>ans) ans=j-i+1,ansl=i,ansr=j;
}
printf("%d %d\n",ansl,ansr);
} signed main(void){
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
#endif
scanf("%d%d%d",&n,&k,&d);
for(int i=1;i<=n;i++) scanf("%d",&a[i]);
for(int i=1;i<=n;i++) a[i]+=1e9;ans=ansl=ansr=1;
if(d==0) return solve(),0;
for(int i=1;i<=n;i++){
if(mp.find(a[i])==mp.end()) mp[a[i]]=i,pre[i]=0;
else pre[i]=mp[a[i]],mp[a[i]]=i;
}
for(int i=1;i<=n;i++) b[i]=a[i]%d,c[i]=a[i]/d;
build(1,n,1);
for(int i=n;i>=1;i--){
while(top&&c[stk[top]]>=c[i]) lmin[stk[top--]]=i;
stk[++top]=i;
}
while(top) lmin[stk[top--]]=0;
for(int i=n;i>=1;i--){
while(top&&c[stk[top]]<=c[i]) lmax[stk[top--]]=i;
stk[++top]=i;
}
while(top) lmax[stk[top--]]=0;
for(int i=1,j;i<=n;i=j+1){
for(j=i;j<n&&b[j]==b[j+1];j++);
for(int t=i,tmp;t<=j;t++){
if(pre[t]>=i){
j=pre[t];
break;
}
change(max(i,lmax[t]+1),t,0,c[t],1),
change(max(i,lmin[t]+1),t,1,c[t],1);
tmp=query(i,t,1,k+t);
if(t-tmp+1>ans) ans=t-tmp+1,ansl=tmp,ansr=t;
}
}
printf("%d %d\n",ansl,ansr);
return 0;
}
By NeighThorn
BZOJ 4527: K-D-Sequence的更多相关文章
- [bzoj P4504] K个串
[bzoj P4504] K个串 [题目描述] 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次 ...
- BZOJ 3110 K大数查询 | 整体二分
BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...
- 【BZOJ 4527】 4527: K-D-Sequence (线段树)
4527: K-D-Sequence Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 145 Solved: 59 Description 我们称一个 ...
- BZOJ 3110 k大数查询 & 树套树
题意: 有n个位置,每个位置可以看做一个集合,现在要求你实现一个数据结构支持以下功能: 1:在a-b的集合中插入一个数 2:询问a-b集合中所有元素的第k大. SOL: 调得火大! 李建说数据结构题能 ...
- [BZOJ]3110 K大数查询(ZJOI2013)
这大概是唯一一道小C重写了4次的题目. 姿势不对的树套树(Fail) → 分块(Fail) → 整体二分(Succeed) → 树套树(Succeed). 让小C写点心得静静. Description ...
- bzoj : 4504: K个串 区间修改主席树
4504: K个串 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 268 Solved: 110[Submit][Status][Discuss] ...
- BZOJ - 4520 K远点对
题意:已知平面内 N 个点的坐标,求欧氏距离下的第 K 远点对 维护大小为2k最小堆,KD树的估值用前面提到的做法 PS.网上有人估价是使用边界四个点的最值来独立枚举,然而这样写似乎过不了 #incl ...
- bzoj 1005 组合数学 Purfer Sequence
这题需要了解一种数列: Purfer Sequence 我们知道,一棵树可以用括号序列来表示,但是,一棵顶点标号(1~n)的树,还可以用一个叫做 Purfer Sequence 的数列表示 一个含有 ...
- bzoj 2656 [Zjoi2012]数列(sequence) 递推+高精度
2656: [Zjoi2012]数列(sequence) Time Limit: 2 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Descri ...
随机推荐
- HTML布局的元素
header 定义文档或节的页眉 nav 定义导航链接的容器 section 定义文档中的节 article 定义独立的自包含文章 aside 定义内容之外的内容(比如侧栏) footer 定义文档或 ...
- 安装一个apk文件源代码
/** * 安装一个apk文件 * * @param file * 要安装的完整文件名 */ protected void installApk(File file) { ...
- Qt BarChart实践
按照帮助文档编写 运行截图 上代码 #include "widget.h" #include "ui_widget.h" Widget::Widget(QWid ...
- 用Python 的一些用法与 JS 进行类比,看有什么相似?
Python 是一门运用很广泛的语言,自动化脚本.爬虫,甚至在深度学习领域也都有 Python 的身影.作为一名前端开发者,也了解 ES6 中的很多特性借鉴自 Python (比如默认参数.解构赋值. ...
- 梳理 Opengl ES 3.0 (一)宏观着眼
Opengl ES 可以理解为是在嵌入式设备上工作的一层用于处理图形显示的软件,是Opengl 的缩水版本. 下图是它的工作流程示意图: 注意图中手机左边的EGL Layer Opengl ES是跨平 ...
- 分布式文件系统---GlusterF
1.1 分布式文件系统 1.1.1 什么是分布式文件系统 相对于本机端的文件系统而言,分布式文件系统(英语:Distributed file system, DFS),或是网络文件系统(英语:Ne ...
- 安装arch系统时,把ubuntu的efi分区格式化
导致无法从grub进入ubuntu,之后我进入win10,把ubuntu的分区都删了. 再重启,只能进入黑色的grub界面,显示 grub>> 甚至无法进入win10.只能通过在开机时按F ...
- 第十三次ScrumMeeting会议
第十三次Scrum Meeting 时间:2017/12/1 地点:咖啡馆 人员:策划组美工组 名字 完成的工作 计划工作 蔡帜 完成公式的基本策划,Bug数量产生机制设计 科技树方面机制确定 游心 ...
- MySQL 初识01
最近开始学习MySQL 所以将这两天所学习到的知识简单小结一下 1.status 显示数据库信息 2.数据类型: a.字符串: char(m):固定长度的字符,最多255个字符: varchar(m) ...
- java文件的I/O
[原创] java文件的I/O操作,简单来说就是向文件中写入数据以及从文件中读出数据,这是我们平日做的最多的操作,这里给出两种文件I/O操作,当然还有许多的操作方法,各种流的使用,可谓是高深莫测:不管 ...