代码随想录算法训练营第三十三天 | 322. 零钱兑换 279.完全平方数 139.单词拆分

无论排列还是组合,元素个数是一样的。

背包问题中 dp[j] 的含义都是装满容量为 j 的背包,所以如果装不满,dp[j]还是初始值。

Go 复制代码
import "math"

func coinChange(coins []int, amount int) int {
	dp := make([]int, amount+1)
	dp[0] = 0
	for i := 1; i <= amount; i++ {
		dp[i] = math.MaxInt32
	}
	for _, coin := range coins {
		for j := coin; j <= amount; j++ {
			dp[j] = min(dp[j], dp[j-coin]+1)
		}
	}
	if dp[amount] == math.MaxInt32 {
		return -1
	}
	return dp[amount]
}

本题 和 322. 零钱兑换 基本是一样的

Go 复制代码
func numSquares(n int) int {
	dp := make([]int, n+1)
	dp[0] = 0
	for i := 1; i <= n; i++ {
		dp[i] = n + 1
	}
	for i := 1; i*i <= n; i++ {
		for j := i * i; j <= n; j++ {
			dp[j] = min(dp[j], dp[j-i*i]+1)
		}
	}
	return dp[n]
}
  1. 也可以算作回溯算法中的分割问题,再用记忆化递归优化。
  2. 前面的题要么是标准背包(求max价值),要么是多少种方法,要么是最少的物品个数。本题的状态转移方程就很灵活,也可以不理解成背包问题,只是一个普通的动态规划问题。
  3. s是物品的排列,所以要先遍历背包。

视频为什么说 可以不理解成背包问题,我认为是代码没有体现出来,如果按我这样写就是很明显的背包问题代码;答案代码也很好,只是没明显体现出是背包问题。

Go 复制代码
func wordBreak(s string, wordDict []string) bool {
	dp := make([]bool, len(s)+1)
	dp[0] = true
	for j := 1; j <= len(s); j++ {
		for _, word := range wordDict {
			if j >= len(word) {
				if dp[j-len(word)] && s[j-len(word):j] == word {
					dp[j] = true
					break
				}
			}
		}
	}
	return dp[len(s)]
}

其中状态转移方程是

Go 复制代码
				if dp[j-len(word)] && s[j-len(word):j] == word {
					dp[j] = true
					break
				}

和答案代码最主要的差别就是遍历了物品(word)而不是s中的下标 i

相关推荐
Yzzz-F1 小时前
Problem - 2205D - Codeforces
算法
智者知已应修善业2 小时前
【51单片机2个按键控制流水灯运行与暂停】2023-9-6
c++·经验分享·笔记·算法·51单片机
Halo_tjn2 小时前
Java Set集合相关知识点
java·开发语言·算法
生成论实验室3 小时前
《事件关系阴阳博弈动力学:识势应势之道》第四篇:降U动力学——认知确定度的自驱演化
人工智能·科技·神经网络·算法·架构
AI科技星3 小时前
全域数学·72分册:场计算机卷【乖乖数学】
算法·机器学习·数学建模·数据挖掘·量子计算
科研前沿4 小时前
镜像孪生VS视频孪生核心技术产品核心优势
大数据·人工智能·算法·重构·空间计算
水蓝烟雨4 小时前
1931. 用三种不同颜色为网格涂色
算法·leetcode
晨曦夜月4 小时前
map与unordered_map区别
算法·哈希算法
图码5 小时前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻
handler015 小时前
Linux 内核剖析:进程优先级、上下文切换与 O(1) 调度算法
linux·运维·c语言·开发语言·c++·笔记·算法