好久没更题解了(改题困难的我)

题目描述

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的刷子的更多相关文章

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

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

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

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

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

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

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

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

  5. bzoj1510: [POI2006]Kra-The Disks(单调栈)

    这道题可以O(n)解决,用二分还更慢一点 维护一个单调栈,模拟掉盘子的过程就行了 #include<stdio.h> #include<string.h> #include&l ...

  6. BZOJ1057[ZJOI2007]棋盘制作 [单调栈]

    题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳. 而我们的 ...

  7. 洛谷U4859matrix[单调栈]

    题目描述 给一个元素均为正整数的矩阵,上升矩阵的定义为矩阵中每行.每列都是严格递增的. 求给定矩阵中上升子矩阵的数量. 输入输出格式 输入格式: 第一行两个正整数n.m,表示矩阵的行数.列数. 接下来 ...

  8. POJ3250[USACO2006Nov]Bad Hair Day[单调栈]

    Bad Hair Day Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 17774   Accepted: 6000 Des ...

  9. CodeForces 548D 单调栈

    Mike and Feet Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Subm ...

随机推荐

  1. 从linux源码看socket的阻塞和非阻塞

    从linux源码看socket的阻塞和非阻塞 笔者一直觉得如果能知道从应用到框架再到操作系统的每一处代码,是一件Exciting的事情. 大部分高性能网络框架采用的是非阻塞模式.笔者这次就从linux ...

  2. Linux权限管理命令chown、chgrp、umask详解

    命令chown详解 命令chown,所在路径为: 可以看到,这个命令的路径为:/usr/bin/chown ,所以它的执行权限是所有用户 命令的基本功能是改变文件或目录的所有者(只有root可以进行, ...

  3. Python学习之输入输出入门 A+B篇

    描述 求两个整数之和. 输入 输入数据只包括两个整数A和B. 输出 两个整数的和. 样例输入 1 2 样例输出  3 a=input().split() print(int(a[0])+int(a[1 ...

  4. 「持续集成实践系列 」Jenkins 2.x 构建CI自动化流水线常见技巧

    在上一篇文章中,我们介绍了Jenkins 2.x实现流水线的两种语法,以及在实际工作中该如何选择脚本式语法或声明式语法.原文可查阅:「持续集成实践系列」Jenkins 2.x 搭建CI需要掌握的硬核要 ...

  5. 因为 MongoDB 没入门,我丢了一份实习工作

    有时候不得不感慨一下,系统升级真的是好处多多,不仅让我有机会重构了之前的烂代码,也满足了我积极好学的虚荣心.你看,Redis 入门了.Elasticsearch 入门了,这次又要入门 MongoDB, ...

  6. v-else-if(v-show)

    <div id="app"> <div v-if="type === 'A'"> A </div> <div v-el ...

  7. Clear Writer v1.8 更新

    拖更了这么久之后,Clear Writer 诈尸啦(bushi 下载链接:https://linhongping.lanzous.com/ikF2Udmf7if Clear Writer v1.8 更 ...

  8. (一)Log4j使用

    原文链接:https://www.jianshu.com/p/eb4ac2571c94?tdsourcetag=s_pctim_aiomsg 1.先创建个maven项目,在我们项目的pom文件中导入l ...

  9. frp多层socks代理+端口映射

    一.首先在公网上配置服务端(frps.ini) [common] bind_addr = xx.xx.xx.xx #公网vps的ip bind_port = 7000   二.配置客户端frpc. i ...

  10. hdoj3791

    题目: Problem Description 判断两序列是否为同一二叉搜索树序列   Input 开始一个数n,(1<=n<=20) 表示有n个需要判断,n= 0 的时候输入结束.接下去 ...