题意

给定一个序列,求有多少个最长连续子序列满足最大值减最小值之差不超过\(k\)。

分析

  • 跟序列最大值最小值有关的可以想到单调栈,先预处理出每个数作为最大值能延伸的区间,然后枚举每个数作为最大值。
  • 最大的满足条件的连续序列显然左边就是要在\([le[i],i-1]\)里找到大于等于\(a[i]-k\)的最小值对应的下标,右边同理。
  • 线段树维护区间最小值,然后再套一个二分(应该也可以不用,就一个log,不过二分容易想),求出两端能满足条件的序列端点,计算出长度更新答案。

代码

#include <bits/stdc++.h>
using namespace std;
#define ls i<<1
#define rs i<<1|1
#define mid (l+r)/2
const int N=1e5+50;
int n,k,a[N];
int le[N],ri[N];
int mn[N*4];
void build(int i,int l,int r){
if(l==r){
mn[i]=a[l];
return;
}
build(ls,l,mid);
build(rs,mid+1,r);
mn[i]=min(mn[ls],mn[rs]);
}
int query(int i,int l,int r,int ql,int qr){
if(ql<=l && qr>=r){
return mn[i];
}
int ans=0x3f3f3f3f;
if(ql<=mid){
ans=min(ans,query(ls,l,mid,ql,qr));
}
if(qr>mid){
ans=min(ans,query(rs,mid+1,r,ql,qr));
}
return ans;
}
set<pair<int,int> > res;
int mx;
int main(){
// freopen("in.txt","r",stdin);
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++){
scanf("%d",&a[i]);
}
stack<int> s;
for(int i=1;i<=n;i++){
while(!s.empty() && a[s.top()]<=a[i]){
s.pop();
}
if(s.empty()){
le[i]=1;
}else{
le[i]=s.top()+1;
}
s.push(i);
}
while(!s.empty()){
s.pop();
}
for(int i=n;i>=1;i--){
while(!s.empty() && a[s.top()]<=a[i]){
s.pop();
}
if(s.empty()){
ri[i]=n;
}else{
ri[i]=s.top()-1;
}
s.push(i);
}
build(1,1,n);
for(int i=1;i<=n;i++){
int t=a[i]-k;
//查询[le[i],i-1] >=t 的最小值的值和位置
int L=le[i],R=i-1;
int lidx=i;
while(L<=R){
int M=(L+R)/2;
int tm=query(1,1,n,M,i-1);
if(tm>=t){
lidx=M;
R=M-1;
}else{
L=M+1;
}
}
L=i+1,R=ri[i];
int ridx=i;
while(L<=R){
int M=(L+R)/2;
int tm=query(1,1,n,i+1,M);
if(tm>=t){
ridx=M;
L=M+1;
}else{
R=M-1;
}
}
int ans=ridx-lidx+1;
if(ans>mx){
mx=ans;
res.clear();
res.insert({lidx,ridx});
}else if(ans==mx){
res.insert({lidx,ridx});
}
}
int siz=res.size();
printf("%d %d\n",mx,siz);
for(auto it:res){
printf("%d %d\n",it.first,it.second);
}
return 0;
}

Codeforces6E_Exposition的更多相关文章

随机推荐

  1. Sqlmap自动注入--REQEST

    数据段: --data sqlmap.py -u ”192.168.1.101/mullitea/login.php" --data="username&passwd=2& ...

  2. 【Python】学习笔记一:Hello world

    前言 在我看来,无论我们学习什么语言第一个学习的估计都是Hello world,那么接下来就从Hello world说起! 编写代码 我在本机上已经安装了pycharm,所以我所编辑的代码都是在pyc ...

  3. Kaggle 网站上的练习

    快速的数据科学教程: https://www.kaggle.com/learn/overview

  4. mysql 判断时间 语法

    今天  select * from 表名 where to_days(时间字段名) = to_days(now());  昨天  SELECT * FROM 表名 WHERE TO_DAYS( NOW ...

  5. python利器之切片

     切片 切片的语法表达式为:[start_index : end_index : step],其中: start_index表示起始索引 end_index表示结束索引 step表示步长,步长不能为0 ...

  6. 后盾网lavarel视频项目---模型一对多关联简单实例

    后盾网lavarel视频项目---模型一对多关联简单实例 一.总结 一句话总结: 在模型中定义一个方法来设置一对多关联:return $this->hasMany(Video::class); ...

  7. node 常用模块

    像在服务器上的操作,我们只要 require 引入的模块,只要不是 nodejs 中的模块,我们的下载环境都是开发环境 配置自动化:引用插件 nodemon 下载:npm i nodemon -g  ...

  8. SQL Server AlwaysOn原理简介

    SQL Server2012所支持的AlwaysOn技术集中了故障转移群集.数据库镜像和日志传送三者的优点,但又不相同.故障转移群集的单位是SQL实例,数据库镜像和日志传送的单位是单个用户数据库,而A ...

  9. C语言转义字符表和ASCII码表

    主要参考 http://www.51hei.com/mcu/4342.html 以及 https://www.cnblogs.com/jason207489550/p/6663444.html

  10. leetcode 31下一个排列

    /** 验证一般情况(元素数目大于等于3)有几个情况分析:两个特殊情况: 6 5 4 3 2 1 完全反序,这种序列没有下一个排序,因此重新排序1 2 3 4 5 6 1 2 3 4 5 6 完全升序 ...