P5019 铺设道路题解

题目传送门

一、贪心法

题目里给的样例是432535,可以选择一个区间进行“填坑”操作;我们的贪心策略是:
a[i]>a[i1],计数器sum+=a[i]a[i1];

贪心证明

假设现在有一个坑,但旁边又有一个坑。你肯定会选择把两个同时填充,都减1;那么小的坑肯定会被大的坑“带着”填掉。大的坑也会减少a[i]a[i1]的深度,可以说是“免费的”;所以这样贪心是对的。

#include <bits/stdc++.h>

using namespace std;
typedef long long LL;
const int N = 1000010;
int a[N];

LL ans;
int n;

int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) cin >> a[i];
    for (int i = 1; i <= n; i++) if (a[i] > a[i - 1]) ans += a[i] - a[i - 1];
    cout << ans;
    return 0;
}

2、递推法

f[i]表示前i个坑所铺设的最少次数。
那么要做的只需比较一下当前的a[i]就是坑的深度)和a[i1],分两种情况:

1、如果a[i]<=a[i1],那么在填a[i1]时就可以顺便把a[i]填上,这样显然更优,所以f[i]=f[i1];

2、如果a[i]>a[i1],那么在填a[i1]时肯定要尽量把a[i]一块填上一部分(a[i1]),a[i]剩余的就单独填。

所以,f[i]=f[i1]+(a[i]a[i1])
初始化f[1]=a[1],向后推就行了,复杂度大概是O(n)

#include <bits/stdc++.h>

using namespace std;
const int N = 1000010;
//递推法
int n, a[N];
int f[N];   //代表前i个区域被填充好的最少次数.套路,满满的套路

int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) cin >> a[i];
    //递推出口,第一个有多大的坑,不能指望别人,都要靠自己来完成~
    f[1] = a[1];
    //从第2个开始进行递推
    for (int i = 2; i <= n; i++) {
        //递推式:见题解的分析过程
        if (a[i] <= a[i - 1]) f[i] = f[i - 1];
        else f[i] = f[i - 1] + (a[i] - a[i - 1]);
    }
    //输出大吉
    cout << f[n] << endl;
    return 0;
}
posted @   糖豆爸爸  阅读(64)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2018-07-21 在Centos中安装aria2c
2016-07-21 PHP、JAVA、C#、Object-C 通用的DES加密
Live2D
点击右上角即可分享
微信分享提示