题目描述

给出 \(N\) 个正整数 \(a[1..N]\) ,再给出一个正整数 \(k\) ,现在可以进行如下操作:每次选择一个大于 \(k\) 的正整数 \(a[i]\) ,将 \(a[i]\) 减去 \(1\) ,选择 \(a[i-1]\) 或 \(a[i+1]\) 中的一个加上 \(1\) 。经过若干次操作后,问最大能够选出多长的一个连续子序列,使得这个子序列的每个数都不小于 \(k\) .

初步分析

其实这个题与下述题目等效:

给出 \(N\) 个正整数 ( \(a[i]-k\) ) 组成的序列,序列前 \(i\) 项和记为 \(s[i]\) ,求满足 \(s[i]> s[j]\) 且 \(i> j\) 的 \(j-i\) 的最大值.

我们来思考一下为什么两个问题会相等.

容易想到,只有平均数大于等于 \(k\) 的子序列经过操作后能够使其每个数都不小于 \(k\),等效于在 \(a[i]-k\) 构成的序列中求最大的和为非负的区间. 而区间和又等于 \(s[j]-s[i-1]\) ,即 \(s[j]-s[i-1]> 0\),整理得 \(s[j]> s[i-1]\),此时区间长度 \(j-i+1\) 即为答案.

因此,我们现在的问题就转化为,如何求这样的区间最大值.

区间最值

首先放出引理:若 \(s[i_{1}] < s[j],\ s[i_{2}]< s[j],\ i_{1} < i_{2} \le j\),那么 \(i_{2}\) 一定不是最优解.

根据此引理,我们定义有可能为最优解的条件:符合条件,且其左方没有符合条件的数.

所以,如果我们从左边遍历,遇到第一个符合条件的数,那么符合该条件的数只有它可能是最优解,所以我们直接改到下一个条件遍历.

总结出如下遍历步骤:

  • 设定初始条件为 \(s[j]=0\).
  • 从左向右尝试放入前缀和.
  • 当前前缀和满足 \(s[i] > s[j]\),放入,并将 \(s[i]\) 作为新的条件.
  • 不满足,忽略.

前缀和有正有负,那么为什么初始条件是 \(s[j]=0\) 呢. 这是为了防止将正的前缀和放进去. 因为实际上正的前缀和可以直接作为答案区间,不需要进行这步流程,我们把它放进去反而会跑错.

注意到此步骤可以用栈实现:

s.push(0);
for(int i=1;i<=n;++i){
sum[i]=sum[i-1]+a[i]-k;
if(sum[s.top()]>sum[id]){
s.push(id);
}
}

我们通过上述步骤,将栈内的内容维护成了一个严格单调递减的序列的下标,这样的话,只要满足 \(i < j\),就有 \(s[i]<s[j]\).

接下来我们来考虑如何求最大区间.

由上述引理,我们倒序遍历全部前缀和 \(s[i]\) , 设栈顶前缀和为 \(s[top]\),那么我们可以遍历到满足 \(s[top]<s[i]\) 的最后一个数,此时区间长度即为 \(i-top\).

代码实现

stack<long long> s;
long long a[1000001],sum[1000001];
void push(int id){
if(sum[s.top()]>sum[id]){
s.push(id);
}
}
long long calc(int n){
long long ans=0;
for(int i=n;i>=1;--i){
while(!s.empty()&&sum[i]-sum[s.top()]>=0){
ans=max(ans,i-s.top());
s.pop();
}
}
return ans;
}
int main(){
int n,m,k;
cin>>n>>m;
for(int i=1;i<=n;++i){
cin>>a[i];
}
for(int i=1;i<=m;++i){
cin>>k;
s.push(0);
for(int i=1;i<=n;++i){
sum[i]=sum[i-1]+a[i]-k;
push(i);
}
s.push(n);
cout<<calc(n)<<" ";
while(!s.empty()){
s.pop();
}
}
}

[TK] Blocks 单调栈的更多相关文章

  1. bzoj 2086 [Poi2010]Blocks 单调栈

    [Poi2010]Blocks Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 788  Solved: 356[Submit][Status][Dis ...

  2. bzoj2086: [Poi2010]Blocks DP,单调栈

    题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=2086 思路 这就有点妙了 题目意思就是让你求平均数>=k的最长序列 先求出a[i] ...

  3. poj 3415 Common Substrings(后缀数组+单调栈)

    http://poj.org/problem?id=3415 Common Substrings Time Limit: 5000MS   Memory Limit: 65536K Total Sub ...

  4. POJ3415 Common Substrings —— 后缀数组 + 单调栈 公共子串个数

    题目链接:https://vjudge.net/problem/POJ-3415 Common Substrings Time Limit: 5000MS   Memory Limit: 65536K ...

  5. [bzoj2086][Poi2010]Blocks_单调栈_双指针

    Blocks bzoj-2086 Poi-2010 题目大意:题目链接. 注释:略. 想法:首先,不难发现,如果连续的一段数的平均值不小于输入的k的话,这段数是满足题意的. 所以,我们再次简化一下:将 ...

  6. POJ - 3415 Common Substrings(后缀数组求长度不小于 k 的公共子串的个数+单调栈优化)

    Description A substring of a string T is defined as: T( i, k)= TiTi+1... Ti+k-1, 1≤ i≤ i+k-1≤| T|. G ...

  7. BZOJ1012: [JSOI2008]最大数maxnumber [线段树 | 单调栈+二分]

    1012: [JSOI2008]最大数maxnumber Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 8748  Solved: 3835[Submi ...

  8. BZOJ 4453: cys就是要拿英魂![后缀数组 ST表 单调栈类似物]

    4453: cys就是要拿英魂! Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 90  Solved: 46[Submit][Status][Discu ...

  9. BZOJ 3238: [Ahoi2013]差异 [后缀数组 单调栈]

    3238: [Ahoi2013]差异 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2326  Solved: 1054[Submit][Status ...

  10. poj 2559 Largest Rectangle in a Histogram - 单调栈

    Largest Rectangle in a Histogram Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 19782 ...

随机推荐

  1. 新做了一个MySQL 数据库 DDL 差异对比的网站

    MySQL 数据库 DDL 差异对比的网站 摘要 新做了个网站,用来对比不同环境下的 DDL 差异,生成变更点和 迁移 DDL 网站地址:https://ddlcompare.com/ 对比过程中如果 ...

  2. Linux环境 yum,apt-get,rpm,wget 区别

    Linux环境 yum,apt-get,rpm,wget 区别 一般来说linux系统基本上分两大类:cat /etc/issue查看linux系统版本RedHat系列:Redhat.Centos.F ...

  3. telegraf 常用命令总结

    本文为博主原创,转载请注明出处: Telegraf 是一个灵活的服务器代理,用于收集和报告指标.它支持插件驱动,这意味着你可以根据需要添加或修改功能. 1.使用telegraf --help 查看te ...

  4. 决定了,今日起开始准备弃用京东JD

    估计京东是为了节约开支,然后开始大比例的把快递物流业务进行外包了,这直接导致服务质量的直线下滑,10多年前我选择弃用当当网而选择京东JD就是因为当时当地的当当网快递是用沈阳晚报的快递上门的,快递员连P ...

  5. 如何在多台Linux系统主机上实现ssh免密访问——成公钥文件id_rsa.pub(数字签名RSA)

    假设共有三台Linux主机,为matser,slave1,slave2,现在要实现master主机可以ssh免密访问master主机自身以及slave1.slave2. 原理: 主机调用秘钥生成命令, ...

  6. 洛谷P5250 【深基17.例5】木材仓库

    [深基17.例5]木材仓库 题目描述 博艾市有一个木材仓库,里面可以存储各种长度的木材,但是保证没有两个木材的长度是相同的.作为仓库负责人,你有时候会进货,有时候会出货,因此需要维护这个库存.有不超过 ...

  7. StarNet:关于 Element-wise Multiplication 的高性能解释研究 | CVPR 2024

    论文揭示了star operation(元素乘法)在无需加宽网络下,将输入映射到高维非线性特征空间的能力.基于此提出了StarNet,在紧凑的网络结构和较低的能耗下展示了令人印象深刻的性能和低延迟 来 ...

  8. 19-canvas绘制文字

    1 <!DOCTYPE html> 2 <html lang="en"> 3 <head> 4 <meta charset="U ...

  9. 23暑假友谊赛No.2

    23暑假友谊赛No.2 A-雨_23暑假友谊赛No.2 (nowcoder.com) #include <bits/stdc++.h> using namespace std; signe ...

  10. 首次在WebAPI中写单元测试

    xUnit 这次我使用的是xUnit测试框架,而不是VS自带的MSTest框架.在添加新建项目时选择xUnit测试项目就行了. 目前只体验到了一个差别,即xUnit可以使用特性向测试方法传参,而不用在 ...