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

相关推荐
Q741_14717 分钟前
每日一题 力扣 3655. 区间乘法查询后的异或 II 模拟 分治 乘法差分法 快速幂 C++ 题解
c++·算法·leetcode·模拟·快速幂·分治·差分法
The_Ticker18 分钟前
印度股票实时行情API(低成本方案)
python·websocket·算法·金融·区块链
夏乌_Wx22 分钟前
剑指offer | 2.4数据结构相关题目
数据结构·c++·算法·剑指offer·c/c++
AI成长日志1 小时前
【笔面试算法学习专栏】哈希表基础:两数之和与字母异位词分组
学习·算法·面试
abant21 小时前
leetcode 239 单调队列 需要一些记忆
算法·leetcode·职场和发展
漫霂1 小时前
二叉树的统一迭代遍历
java·算法
炽烈小老头1 小时前
【每天学习一点算法 2026/04/08】阶乘后的零
学习·算法
Mr_Xuhhh2 小时前
算法刷题笔记:从滑动窗口到哈夫曼编码,我的算法进阶之路
开发语言·算法
MicroTech20252 小时前
突破虚时演化非酉限制:MLGO微算法科技发布可在现有量子计算机运行的变分量子模拟技术
科技·算法·量子计算
hssfscv2 小时前
软件设计师下午题六——Java的各种设计模式
java·算法·设计模式