力扣198-打家劫舍(Java详细题解)

题目链接:198. 打家劫舍 - 力扣(LeetCode)

前情提要:

因为本人最近都来刷dp类的题目所以该题就默认用dp方法来做。

dp五部曲。

1.确定dp数组和i下标的含义。

2.确定递推公式。

3.dp初始化。

4.确定dp的遍历顺序。

5.如果没有ac打印dp数组 利于debug。

每一个dp题目如果都用这五步分析清楚,那么这道题就能解出来了。

题目思路:

刚接触打家劫舍的朋友们入手此题,可能还是有些难度。

本题偷窃金额只能选择不相邻的俩家偷,那我们怎么选择呢?

只能隔一个偷吗,其实也可以隔好几个,所以此时大家就有些懵了,不知道如何下手。

其实分析每一个房间(子问题)偷或不偷是取决于他前一个房间和前俩个房间的状态。

比如当前房间能偷,那么他上一个房间就不能偷,前俩个房间就可以考虑偷。

如果当前房间不偷,那么他上一个房间就可以考虑偷了。

注意我这里是考虑,并不是一定非要偷这些房间,真正实现偷这些房间是递推公式帮我们实现的,我们只要把该情况考虑进去就可以了。

接下来我们用动规五部曲来系统分析一下。

1.确定dp数组和i下标的含义。

dp[i] 表示的就是考虑i房间之前包括i房间所偷窃的最大金额。

2.确定递推公式。

上面的分析其实已经把递推公式得出来了。

当前房间偷的状态就是 dp[ i - 2] + nums[i]。

当前房子偷,前一个房间肯定是不能偷,只能从他前俩个房间开始考虑偷。

dp[i - 2]就是考虑偷 i - 2之前的房间偷窃的最大金额。

当前不偷的状态就是dp[i - 1]。

当前房子不偷,前一个房间就可以考虑偷。

因为我们是要求最大金额,那么当前房子偷或者不偷的金额我得取一个最大。

所以

dp[i] = Math.max(dp[ i - 2] + nums[i],dp[ i -1]);

3.dp初始化。

由递推公式可以知道dp[i]是由dp[i - 2]和dp[i - 1]得出。

所以dp[0]和dp[1]是递推的入口。

dp[0]初始化为多少呢?

dp[0]表示的就是当前房子下标为0时能偷的最大金额。其实就是nums[0],因为这有这个房间能偷。

dp[1]表示的就是考虑房子下标为1和之前房子能偷的最大金额。那么这里我们也只能偷一个,所以对nums[0]和nums[1]取一个最大偷就好。

4.确定dp的遍历顺序。

从递推公式中也能知道dp[i]是由dp[i - 2]和dp[i - 1]得出。

所以dp的遍历顺序肯定是从前往后的。

5.如果没有ac打印dp数组 利于debug。

建议大家手动模拟一下,便于理解,

在这里插入图片描述

最终代码:

class Solution {
    public int rob(int[] nums) {
        int [] dp = new int [nums.length + 1];
        //dp数组指的是在i房间前(包括i)所偷窃到的最高金额
        //因为这里我们初始化了dp[0]和dp[1]所以当dp.length == 1时就不存在dp[1]就会报错,所以这里我们需要特判一下
        //该题每个房间只有选和不选 选的话肯定上一个就不能选 就得从上上个开始偷窃
        //不选的话 就可以从上一个开始偷窃
        if (nums.length == 1) return nums[0];
        //dp初始化
        dp[0] = nums[0];
        dp[1] = Math.max(nums[0],nums[1]);
        //dp的遍历顺序
        for(int i = 2;i < nums.length;i++){
            dp[i] = Math.max(dp[i - 2] + nums[i],dp[i - 1]);
        }
        return dp[nums.length - 1];
    }
}

这一篇博客就到这了,如果你有什么疑问和想法可以打在评论区,或者私信我。

我很乐意为你解答。那么我们下篇再见!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值