题意

给定一个序列,求有多少个最长连续子序列满足最大值减最小值之差不超过\(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. input输入框如何只能输入非零开头的正整数

    input输入框如何只能输入非零开头的正整数 ********* 废话不多说,先来代码 ********* case1: 原生html + javascript <body> <!- ...

  2. Java虚拟机之垃圾回收算法思想总结

    1.引用计数法 这是个比较古老而经典的垃圾回收算法,其核心就是在对象被其他所引用的时候计数器加1,而当引用失去时减1.这个方法有非常严重的问题:无法此话有理循环引用的情况,还有就是每次进行加减操作比较 ...

  3. R_Studio(关联)对dvdtrans.csv数据进行关联规则分析

    dvdtrans.csv数据:该原始数据仅仅包含了两个字段(ID, Item) 用户ID,商品名称(共30条) #导入arules包 #install.packages("arules&qu ...

  4. python学习之路(15)

    map/reduce Python内建了map()和reduce()函数. 如果你读过Google的那篇大名鼎鼎的论文“MapReduce: Simplified Data Processing on ...

  5. 前端23种js设计模式中参见的7种设计模式的学习

    创建型设计模式是一类处理对象创建的设计模式,通过某种方式控制对象的创建来避免基本对象创建时可能导致设计上的问题或增加设计上的复杂度. 1)工厂模式 class Product { constructo ...

  6. C++入门经典-例5.20-右值引用传递参数

    1:使用字面值(如1.3.15f.true),或者表达式等临时变量作为函数实参传递时,按左值引用传递参数会被编译器阻止.而进行值传递时,将产生一个和参数同等大小的副本.C++11提供了右值引用传递参数 ...

  7. centos6 centos7 配置开机启动服务

    centos 6 :使用chkconfig命令即可. 我们以apache服务为例: #chkconfig --add apache 添加nginx服务 #chkconfig apache on 开机自 ...

  8. LeetCode 14. 最长公共前缀(Longest Common Prefix)

    题目描述 编写一个函数来查找字符串数组中的最长公共前缀. 如果不存在公共前缀,返回空字符串 "". 示例 1: 输入: ["flower","flow ...

  9. 查询redis中没有设置过期时间的key

    #!/bin/sh ## 该脚本用来查询redis集群中,哪些key是没有设置过期时间,对应只需要修改redis的其中一个实例的 host和port ## 脚本会自动识别出该集群的所有实例,并查出对应 ...

  10. 1.4 Navicat Mybatis 占坑

    MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以使用简单的 XML ...