代码随想录算法训练营第三十三天 | 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

相关推荐
liulilittle1 小时前
C++ 17 字符串填充函数(PaddingLeft、PaddingRight)填充左侧、右侧。
c++·算法
AuroraWanderll1 小时前
深入理解C++多态(三):多态的原理——虚函数表机制(上)
c语言·开发语言·数据结构·c++·算法·stl
mit6.8241 小时前
预统计
算法
良木生香1 小时前
【程序设计】P8772 [蓝桥杯 2022 省 A] 求和
c语言·算法·职场和发展·蓝桥杯
SoleMotive.1 小时前
项目中如何排查解决慢接口问题
数据库·redis·算法·缓存
hadage2331 小时前
--- 算法 分割回文串 回溯 + 动态规划预处理 ---
算法·动态规划
Yzzz-F1 小时前
[模板]单调队列/滑动窗口
算法
做怪小疯子2 小时前
LeetCode 热题 100——二叉树——翻转二叉树
算法·leetcode·职场和发展
报错小能手2 小时前
数据结构 带头节点的双向循环链表
数据结构·算法·链表