洛谷 P15800:[GESP202603 六级] 选数 ← 动态规划

【题目来源】
https://www.luogu.com.cn/problem/P15800

【题目描述】

【输入格式】
第一行,一个正整数,表示数组长度。
第二行,n 个正整数 a1, a2, ..., an,表示数组 a。
第三行,n 个正整数 b1, b2, ..., bn,表示数组 b。

【输出格式】
一行,一个整数,表示在满足下标条件的前提下,数组 a 对应下标的整数之和的最大值。

【输入样例】
6
1 1 4 5 1 4
1 2 3 2 1 0

【输出样例】
11

【数据范围】
对于 40% 的测试点,保证 2≤n≤10^3。
对于所有测试点,保证 2≤n≤10^5,0≤ai≤10^9,0≤bi≤n。

【算法分析】
● 闫氏 DP 分析法:https://www.bilibili.com/video/BV1X741127ZM
● 最后一步法:https://www.bilibili.com/video/BV1xb411e7ww
● 解题思路(来源于:https://mp.weixin.qq.com/s/MzuV2CmidkGD5I-EMGS28A
这是一个典型的带跳跃限制的动态规划问题。如果我们当前选择位置 i,那么下一个可选择的位置必须满足 j≥i+b[i]。因此,可以从 i 跳到区间 [i+b[i], n]。
(1)状态定义
设 f[i] 表示当第一个选择的位置是 i 时,能够获得的最大收益。
(2)状态转移
如果选择了 i,下一步可以选任意 j∈[i+b[i], n]。
于是:f[i]=a[i]+max(f[j]) (其中 j≥i+b[i])
如果 i+b[i]>n,说明后面不能再选。此时,有 f[i]=a[i]。
(3)优化
直接枚举 j 会变成 O(n²)。我们可以维护一个后缀最大值数组 mx[i] = max(f[i], f[i+1], ..., f[n])。这样就能 O(1) 得到 max(f[j]),当 j≥i+b[i]=mx[i+b[i]] 时。
于是转移方程优化为:当 i+b[i]<=n时,f[i]=a[i]+mx[i+b[i]]。否则,f[i]=a[i]。
(4)最终答案
因为可以从任意位置开始,答案就是 max(f[1], f[2], ..., f[n])。
复杂度分析:DP 计算 O(n),空间复杂度 O(n)。整体复杂度 O(n),完美通过。

【算法代码】

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

typedef long long LL;
const int N=1e5+5;
int a[N],b[N];
LL f[N],ans;
int n;

int main() {
    ios::sync_with_stdio(0);
    cin.tie(0);
    cout.tie(0);

    cin>>n;
    for(int i=1; i<=n; i++) cin>>a[i];
    for(int i=1; i<=n; i++) cin>>b[i];
    for(int i=1; i<=n; i++) {
        ans=max(ans,f[i]+a[i]);
        if(i+b[i]<=n) {
            f[i+b[i]]=max(f[i+b[i]],f[i]+a[i]);
        }
        f[i+1]=max(f[i+1],f[i]);
    }
    cout<<ans;

    return 0;
}

/*
in:
6
1 1 4 5 1 4
1 2 3 2 1 0

out:
11
*/

【参考文献】
https://gesp.ccf.org.cn/101/attach/1734457386074144.pdf
https://blog.csdn.net/hnjzsyjyj/article/details/113549272
https://mp.weixin.qq.com/s/MzuV2CmidkGD5I-EMGS28A
https://mp.weixin.qq.com/s/xPLBl-FiJOvgF83-lqbGTA

相关推荐
小辉同志19 小时前
139. 单词拆分
算法·动态规划
l1t21 小时前
Qwen 3.5plus一步做对的欧拉计划701题
算法·动态规划·欧拉计划
liuyao_xianhui1 天前
优选算法_两数之和_位运算_C++
java·开发语言·数据结构·c++·算法·链表·动态规划
liuyao_xianhui1 天前
优选算法_位运算_只出现一次的数字3_C++
开发语言·数据结构·c++·算法·leetcode·链表·动态规划
Sakinol#1 天前
Leetcode Hot 100 ——动态规划part01
leetcode·动态规划
We་ct1 天前
LeetCode 53. 最大子数组和:两种高效解法(动态规划+分治)
前端·算法·leetcode·typescript·动态规划·分治
葳_人生_蕤1 天前
hot100——动态规划
算法·动态规划
小范自学编程2 天前
算法训练营 Day37 - 动态规划part06
算法·动态规划
小范自学编程3 天前
算法训练营 Day38 - 动态规划part07
算法·动态规划