算法第三十七天:动态规划part05(第九章)

一.完全背包理论基础-二维DP数组

复制代码
#每个背包可以使用无数次

#倒序遍历改为正序遍历

n, v = map(int, input().split())#读取物品的种类和重量
data = []
for _ in range(n):
    weight, value = map(int, input().split())
    data.append((weight, value))

#初始化 # 初始化 dp[i][j] 表示前 i 个物品在容量为 j 的背包下的最大价值
dp = [[0]*(v+1) for _ in range(n+1)]

for i in range(1,n+1):
    weight, value = data[i-1]
    for j in range(v+1):
        #不选第i个物品
        dp[i][j] = dp[i-1][j]
        #可以选第i个物品多次
        if j >= weight:
            dp[i][j] = max(dp[i][j], dp[i][j-weight]+value)
print(dp[n][v])

2.零钱兑换Ⅱ:组合数:先遍历物品后遍历背包

518. 零钱兑换 II - 力扣(LeetCode)

思路:

复制代码
class Solution:
    def change(self, amount: int, coins: List[int]) -> int:
        #dp[j] 表示可以凑成价值为j的组合数

        dp = [0] * (amount+1)
        dp[0] = 1
        for coin in coins:
            for j in range(coin, amount+1):#循环从 coin 开始的原因:因为会越界

                dp[j] += dp[j-coin]
        return dp[amount]

3.组合总合Ⅳ :排列:先遍历背包再遍历物品

思路:

✅ 思路总结:

📌 1. 本质是一个完全背包问题的变形:
  • 数组元素可以重复使用;

  • 关键在于:组合顺序不同视作不同的结果 ,即 [1,2][2,1] 被认为是两种不同的组合;

  • 所以我们要计算的是排列数(而非组合数)。

2. 状态定义:
  • 定义 dp[i] 表示:凑出目标和为 i 的组合(排列)数
📌 3. 初始化:

dp = [0] * (target + 1) dp[0] = 1 # 凑出 0 有 1 种方式:什么都不选

  • 为什么 dp[0]=1?表示凑成 0 元,有 1 种方式(空序列),这是构建递推关系的基础。

📌 4. 状态转移方程(排列 vs 组合的核心区别):

复制代码
class Solution:
    def combinationSum4(self, nums: List[int], target: int) -> int:
        #强调组合顺序即为排列 :先遍历背包,再遍历物品
        #组合数:先遍历物品再遍历背包
        #dp[j] 就是总和为j的元素组合个数
        dp = [0]*(target+1)
        dp[0] = 1

        for i in range(1, target+1):
            for num in nums:
                if i >= num:
                    dp[i] += dp[i-num]
        return dp[target]

4.爬楼梯:不难套公式,属于排列问题,先背包后物品

复制代码
n, m = map(int, input().split())

#这个属于排列问题
#定义dp数组,dp[j] 等于爬j阶楼梯有dp[j]种不同的方法
dp = [0]*(n+1)
dp[0] = 1

#递推公式:先背包后物品
for i in range(1, n+1):
    for step in range(1, m+1):
        if i >= step:
            dp[i] += dp[i-step]
print(dp[n])

今天就到这里结束啦!明天继续!

相关推荐
stolentime6 分钟前
树套树+标记永久化:[POI 2006] TET-Tetris 3D&&SPOJ1741 TETRIS3D - Tetris 3D题解
c++·算法·线段树·树套树·标记永久化
XiYang-DING7 分钟前
【LeetCode】链表 + 快慢指针找倒数结点 | 链表中倒数第k个结点
算法·leetcode·链表
一轮弯弯的明月21 分钟前
有序整数对个数-欧拉函数
java·算法·蓝桥杯·学习心得
dazzle35 分钟前
机器学习算法原理与实践-入门(十):基于PaddlePaddle框架的线性回归
算法·机器学习·paddlepaddle
2501_9403152637 分钟前
【无标题】1.用哈希表做两数之和
算法·哈希算法·散列表
计算机安禾1 小时前
【数据结构与算法】第20篇:二叉树的链式存储与四种遍历(前序、中序、后序、层序)
c语言·开发语言·数据结构·c++·学习·算法·visual studio
菜菜的顾清寒1 小时前
力扣HOT100(16)除了自身以外数组的乘积
算法·leetcode·职场和发展
sali-tec1 小时前
C# 基于OpenCv的视觉工作流-章45-圆卡尺
图像处理·人工智能·opencv·算法·计算机视觉
会编程的土豆1 小时前
【数据结构与算法】二叉树遍历 集合
数据结构·算法
老虎06271 小时前
LeetCode热题100 刷题笔记(第六天)双指针 「 盛最多水的容器」
笔记·算法·leetcode