算法 198. 打家劫舍 ---------动态规划学习

题目: https://leetcode.cn/problems/house-robber

看到"最高金额" 立即知道求最值一般使用动态规划,这是很经典的动态规划。

dpi 数组的意思是 以 i 结尾的数组可以得到的最高金额,此时小偷可以选择偷 numsi 也可以选择不偷,怎么样可以得到 dpi 值呢?要么是 dpi-2,要么是 dpi-1,其实小偷还可以选择dpi-3,dpi-4....一直到 dp0,但是选择dpi-3,dpi-4...这些数字很明显不可能比 dpi-1和 dpi-2 中的更大,因为 dpi-1 和 dpi-2 就是从 dpi-3,dpi-4,dpi-5......中得来的。

所以在 numsi 位置处的 dpi 要想最大,只可能是 max(dpi-2+numsi, dpi-1) 。其中边界情况和起始条件是,如果nums 数值只有一个值,那就直接返回它,如果有两个值,就返回两者中最大的那个。

Go 复制代码
func rob(nums []int) int {
	var res int
	if len(nums) == 0 {
		return 0
	}
	if len(nums) == 1 {
		res = nums[0]
		return res
	}
	dp := make([]int, len(nums))
	dp[0] = nums[0]
	dp[1] = max(nums[0], nums[1])
	for i := 2; i < len(nums); i++ {
		dp[i] = max(dp[i-2]+nums[i], dp[i-1])
		res = max(res, dp[i])
	}
	return dp[len(nums)-1]
}

213. 打家劫舍 II

https://leetcode.cn/problems/house-robber-ii

这个问题对上面的问题加了限制,即比较nums0,len(nums)-1 和 nums1:len(nums) 的能偷到的最大值。

Go 复制代码
func rob(nums []int) int {
	if len(nums) == 0 {
		return 0
	}
	if len(nums) == 1 {
		return nums[0]
	}
	return max(rob2(nums[0:len(nums)-1]), rob2(nums[1:len(nums)]))
}
func rob2(nums []int) int {
	var res int
	n := len(nums)
	if n == 0 {
		return 0
	}
	if n == 1 {
		res = nums[0]
		return res
	}
	dp := make([]int, n)
	dp[0] = nums[0]
	dp[1] = max(nums[0], nums[1])
	for i := 2; i < n; i++ {
		dp[i] = max(dp[i-2]+nums[i], dp[i-1])
		res = max(res, dp[i])
	}
	return dp[n-1]
}
相关推荐
J2虾虾5 分钟前
C语言 typedef 用法
c语言·数据结构·算法
hunterkkk(c++)13 分钟前
线段树例题
算法
故渊at24 分钟前
第二板块:Android 四大组件标准化学理 | 第七篇:Activity 页面载体与任务栈算法
android·算法·生命周期·activity·任务栈
兰令水31 分钟前
leecodecode【区间DP+树形DP】【2026.6.10打卡-java版本】
java·算法·leetcode
weixin199701080161 小时前
[特殊字符] 1688开放平台API Sign签名算法详解(Java / Python / PHP 实现)
java·python·算法
未若君雅裁1 小时前
JVM 垃圾回收算法与分代回收机制
java·jvm·算法
智者知已应修善业1 小时前
【51单片机初始化D5-D8亮,每按键按下D1到D4全亮,再按下恢复,如此循环】2024-3-26
c++·经验分享·笔记·算法·51单片机
8Qi82 小时前
LeetCode 4:寻找两个正序数组的中位数 —— 二分查找法
java·算法·leetcode·职场和发展·二分查找
8Qi82 小时前
LeetCode 32:最长有效括号 —— 栈 + 标记法 题解
java·数据结构·算法·leetcode·职场和发展··括号匹配