当前位置:首页 > 洛谷 > 洛谷P1616题解:动态规划之完全背包问题

洛谷P1616题解:动态规划之完全背包问题

5个月前 (07-14)

洛谷P1616题解:动态规划之完全背包问题 洛谷题解 动态规划 完全背包问题 C++ 第1张

一、题目解读

洛谷P1616题要求在一个包含M个活动(每个活动有固定时间和价值)的场景中,求解在总时间T内选择活动的最优组合,使得总价值最大化。活动可重复选择,需利用动态规划算法找到最优解。题目强调时间限制与价值收益的平衡,属于典型的完全背包问题变体。

二、解题思路

1. 动态规划核心思想:将问题拆解为子问题,通过状态转移方程逐步求解全局最优解。

2. 完全背包模型:与01背包不同,完全背包允许每个物品(活动)被选择多次,因此状态转移需考虑重复选择的可能性。

3. 优化策略:使用一维DP数组(空间优化),通过正序遍历容量(时间)实现状态更新,避免重复计算。

三、解题步骤

1. 数据输入与初始化:

○ 读取总时间T和活动数量M。

○ 使用vector存储每个活动的时间time和价值value。

○ 初始化DP数组dp,其中dp[j]表示在时间j内可获得的最高价值。

2. 动态规划循环:

○ 外层遍历活动(物品),内层正序遍历时间(容量)。

○ 状态转移方程:dp[j] = max(dp[j], dp[j - time[i]] + value[i]),即当前时间j的价值可选择“不选该活动”或“选该活动并扣除其时间后剩余价值+当前价值”。

3. 输出结果:dp数组的最后一个元素dp[t]即为最优解。

四、代码与注释

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main() {
    int t, m;  // t为总时间,m为活动数量
    cin >> t >> m;
    
    vector<int> time(m), value(m);  // time存储活动时间,value存储活动价值
    for (int i = 0; i < m; ++i) {
        cin >> time[i] >> value[i];
    }
    
    vector<long long> dp(t + 1, 0);  // dp[i]表示前i时间内的最大价值,初始化为0
    
    // 完全背包核心算法
    for (int i = 0; i < m; ++i) {  // 遍历每个活动
        for (int j = time[i]; j <= t; ++j) {  // 正序遍历时间(容量)
            dp[j] = max(dp[j], dp[j - time[i]] + value[i]);  // 状态转移:选择当前活动或不选择
        }
    }
    
    cout << dp[t] << endl;  // 输出最终结果
    return 0;
}

五、总结

本文通过动态规划方法解决了洛谷P1616题中的时间价值优化问题,利用完全背包模型允许重复选择的特点,通过简洁的状态转移方程实现了高效求解。代码采用一维DP数组降低空间复杂度,正序遍历确保重复选择的有效性。该解法为同类背包问题提供了通用框架,需重点理解状态定义与转移逻辑,适用于资源可重复利用的最优决策场景。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣35:二分法在搜索插入位置中的运用

力扣35:二分法在搜索插入位置中的运用

有序数组的定位在一个严格递增的数字序列中,每个元素都有其确定的位置。当新元素试图加入时,我们需要回答两个问题:它是否已经存在?如果不存在,它应该插入在哪里?这道题要求我们在O(log n)时间内完成这...

力扣654:递归分治的艺术 如何用最大元素构建二叉树

力扣654:递归分治的艺术 如何用最大元素构建二叉树

题目重解我们面对一个看似简单却充满递归魅力的题目:给定一个不含重复元素的整数数组,需要构建一棵特殊的二叉树。这个树的每个父节点都必须是当前子数组中的最大元素,而它的左右子树则分别由该最大值左侧和右侧的...

NOIP2005 普及组 洛谷P1408 背包问题的空间优化技巧与实战应用

NOIP2005 普及组 洛谷P1408 背包问题的空间优化技巧与实战应用

题目重解想象你是一名药师,有t分钟在山上采集m种草药。每种草药需要time分钟采集,价值为num。这就像考试时分配时间做题,要选择收益最大的题目组合。题目要求计算在规定时间内能获得的最大草药价值。解题...

IOI 1994 洛谷1216:如何用O(1)空间解决数字三角形问题?附代码实现

IOI 1994 洛谷1216:如何用O(1)空间解决数字三角形问题?附代码实现

题目重解:数字三角形是一个经典的动态规划问题,给定一个由数字组成的三角形结构,从顶部出发,每次可以移动到下方相邻的数字,最终到达底部。我们需要找到一条路径,使得路径上经过的数字总和最大。这个问题可以很...

手搓顺序表类代码注释与详解:从零实现动态数组(新手教程)

一、简介和特点顺序表(Sequential List)是数据结构中基础的一种线性表,其特点是将数据元素存储在连续的内存空间中。通过数组实现,支持随机访问(即通过索引直接访问元素),适用于频繁随机读取的...

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

一、题目解读2024年蓝桥杯省B组“传送阵”题目要求处理一个包含n个节点的图,节点间存在单向传输关系。每个节点i可传送至a[i]指定的节点,形成可能存在的环结构。题目需求解从任意节点出发能到达的最长路...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。