单调栈之WYT的刷子
好久没更题解了(改题困难的我)
题目描述
WYT有一把巨大的刷子,刷子的宽度为M米,现在WYT要使用这把大刷子去粉刷有N列的栅栏(每列宽度都为1米;每列的高度单位也为米,由输入数据给出)。
使用刷子的规则是:
- 与地面垂直,从栅栏的底部向上刷
- 每次刷的宽度为M米(当剩余栅栏宽度不够M米的话,刷子也可以使用,具体看样例2)
- 对于连续的M列栅栏,刷子从底向上,刷到的高度只能到这M列栅栏的最低高度。
WYT请你回答两个问题:
- 最少有多少个单位面积不能刷到(单位面积为1平米)
- 在满足第一问的条件下,最少刷几次?
输入格式
共两行:
第一行两个整数N和M。
第二行共N个整数,表示N列栅栏的高度
输出格式
一行,两个整数,分别为最少剩余的单位面积数量和最少刷的次数。
样例
输入
5 3
5 3 4 4 5
输出
3
2
样例对应解释
高度分别为 5 3 4 4 5 如上:
黄色的方块表示共有3个单位面积没刷上
绿色的框和红色的框表示一共刷了两次。
数据范围与提示
30%的数据:N<=10^3
50%的数据:N<=10^5
100%的数据:1<=N<=10^6, 1<=M<=10^6,N>=M, 每列栅栏的高度<=10^6.
思路
- 这道题当时真的没有想出来,直接暴力模拟拿了点分,这道题的核心在于维护单调栈,用于求解每块木板向右(向左)延伸的长度,拿向右来说,我们维护单调递增的栈,当栈顶元素的长度小于要放入的元素,则正常入栈,top++,如果栈不为空且栈顶元素(假设为sta[top])大于扫描到的元素(假设为i),对栈顶元素执行出栈操作,然后可得sta[top]向右延伸的距离为r[sta[top]]=i-sta[top],r数组维护向右延伸的长度,同理可求得左边延伸长度;
- 处理完l和r数组后,对于木板i,其延伸距离为l[i]+r[i]-1,判断其与刷子宽度的大小,大于则可以刷到,用flag标记,小于则不能。然后枚举每一块木板,维护mh数组为第i块木板能被刷到的长度,如果被标记过,则mh[i]=h[i],如果不能,则mh[i]=mh[i-1],用长度和减去可以被刷到的部分即为ans1;
- 对于每一部分(能被刷到的的长度相等),贪心处理,用其宽度除以刷子宽度,累加得ans2;
附上代码一份
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e6+10;
long long v[maxn],mh[maxn],sum,r[maxn],l[maxn];
long long n,m;
bool flag[maxn];
inline int read(){
int s=0;
char ch=getchar();
while(ch<'0'||ch>'9')ch=getchar();
while(ch>='0'&& ch<='9')s=(s<<3)+(s<<1)+(ch^48),ch=getchar();
return s;
}
long long sta[maxn],top,h[maxn];
int main(){
n=read(),m=read();
for(int i=1;i<=n;i++){
h[i]=read();
sum+=h[i];
}
for(int i=1;i<=n+1;i++){
while(top && h[i]<h[sta[top]]){
r[sta[top]]=i-sta[top];
top--;
}
sta[++top]=i;
}
for(int i=n;i>=0;i--){
while(top && h[i]<h[sta[top]]){
l[sta[top]]=sta[top]-i;
top--;
}
sta[++top]=i;
}
for(int i=1;i<=n;i++){
if(l[i]+r[i]-1>=m)flag[i]=true;
}
for(int i=1;i<=n;i++){
if(flag[i])mh[i]=h[i];
else{
mh[i]=mh[i-1];
}
}
for(int i=n;i>0;i--){
if(!flag[i])mh[i]=max(mh[i],mh[i+1]);
sum-=mh[i];
}
cout<<sum<<endl;
long long k=2,ans=0;
while(k<=n+1){
int cnt=1;
while(k<=n+1 && mh[k]==mh[k-1]){
++cnt;
++k;
}
ans+=cnt/m;
if(cnt%m)++ans;
++k;
}
cout<<ans<<endl;
}
单调栈之WYT的刷子的更多相关文章
- BZOJ1012: [JSOI2008]最大数maxnumber [线段树 | 单调栈+二分]
1012: [JSOI2008]最大数maxnumber Time Limit: 3 Sec Memory Limit: 162 MBSubmit: 8748 Solved: 3835[Submi ...
- BZOJ 4453: cys就是要拿英魂![后缀数组 ST表 单调栈类似物]
4453: cys就是要拿英魂! Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 90 Solved: 46[Submit][Status][Discu ...
- BZOJ 3238: [Ahoi2013]差异 [后缀数组 单调栈]
3238: [Ahoi2013]差异 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2326 Solved: 1054[Submit][Status ...
- poj 2559 Largest Rectangle in a Histogram - 单调栈
Largest Rectangle in a Histogram Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19782 ...
- bzoj1510: [POI2006]Kra-The Disks(单调栈)
这道题可以O(n)解决,用二分还更慢一点 维护一个单调栈,模拟掉盘子的过程就行了 #include<stdio.h> #include<string.h> #include&l ...
- BZOJ1057[ZJOI2007]棋盘制作 [单调栈]
题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳. 而我们的 ...
- 洛谷U4859matrix[单调栈]
题目描述 给一个元素均为正整数的矩阵,上升矩阵的定义为矩阵中每行.每列都是严格递增的. 求给定矩阵中上升子矩阵的数量. 输入输出格式 输入格式: 第一行两个正整数n.m,表示矩阵的行数.列数. 接下来 ...
- POJ3250[USACO2006Nov]Bad Hair Day[单调栈]
Bad Hair Day Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 17774 Accepted: 6000 Des ...
- CodeForces 548D 单调栈
Mike and Feet Time Limit:1000MS Memory Limit:262144KB 64bit IO Format:%I64d & %I64u Subm ...
随机推荐
- SQL手工注入绕过过滤
1.考虑闭合:单引号 --> %27 空格-->%20 井号--> %23 : 构造闭合函数 %27teacher%23 2.判断过滤内容:union --> uniu ...
- 快速幂解法--x^n
class Solution{ public: double myPow(double x,int n){ if(==x || n==) return ; if(n == ) return x; if ...
- Error:org.gradle.api.internal.tasks.DefaultTaskInputs$TaskInputUnionFileCollection cannot be cast to...异常处理
这个是打开Android Studio项目报的错误提示,单纯从上面的提示还是不能太直接的知道什么问题.后来我想这个项目的Gradle版本与我当前AS使用的版本不一致,可能是这个问题. 修改build. ...
- CentOS7——配置阿里云镜像源
CentOS7--配置阿里云镜像源 #下载CentOS 7的repo文件 curl -o /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun ...
- 基于 abp vNext 和 .NET Core 开发博客项目 - Blazor 实战系列(五)
系列文章 基于 abp vNext 和 .NET Core 开发博客项目 - 使用 abp cli 搭建项目 基于 abp vNext 和 .NET Core 开发博客项目 - 给项目瘦身,让它跑起来 ...
- Water Testing【皮克定理,多边形面积,线段上点的数目】
Water Testing 传送门:链接 来源:UPC 9656 题目描述 You just bought a large piece of agricultural land, but you n ...
- 全网最完整的Redis入门指导
前言 本文提供全网最完整的Redis入门指导教程,下面我们从下载Redis安装包开始,一步一步的学习使用. 下载Redis 官网提供的Redis安装包是服务于Linux的,而我们需要在Window下使 ...
- c常用函数-atoi 和 itoa
atoi 和 itoa atoi的功能是把一个字符串转为整数 Action(){ int j; char *s=""; j = atoi(s); lr_output_message ...
- Android学习笔记:实现层级导航
层级导航示例 层级导航案例 1.收下准备两个Activity的布局文件 activity_main.xml <?xml version="1.0" encoding=&quo ...
- 一文搞懂GitLab安装部署及服务配置
GitLab安装部署 Git,GitHub,GitLab,这三个东东长得好像呀,都是个啥? Git是Linus Torvalds(如果不知道这位大神是谁,请出门左转,慢走不送~)为了帮助管理Linux ...