leetcode 198.打家劫舍

首先说知识点,就是动态规划问题。

思路:既然是对于最优方案的考虑,我们就需要有一个思想,就是用动态规划的思想进行思考,动态规划解决的便是这种最优化问题,我们首先想一下,既然说是偷盗最多的金额,那么也就是说,从0-n这些范围中,我们需要逐一判断情况。

当没有房子的时候,自然就是0;当只有一个房子的时候,就只有这一个,也就是nums[0];当我们有两个房子的时候,这个时候就是max(nums[0],nums[1])了;当我们的房子数大于2的时候,我们需要思考:怎样才能不在临近的房子里偷取最多的金额呢?暴力首先是不行的,因为我们如果进行暴力的话还需要回溯,这样写起来会非常麻烦;动态规划首先的一个问题就是在分析当前问题的时候能否选这个房子。OK,我们下面做个假设:

如果说我们现在正准备偷第k间房子,这个时候我们有两个选择:第一个是偷,第二个是不偷。当然,这两种可能我们都需要考虑。有人可能会问了,这第k间房子旁边的房子是否已经偷过了呢?还需要考虑吗?答案是不需要。因为我们从0开始一直在做最优解的结构化,也就是说,我们遍历到k的时候其实前面的k-1间我们已经把该考虑的考虑过了,所以我们只需要考虑当前的选择就行了。

于是,我们分析这两个可能性:第一个,如果我们选择偷的话,那么,旁边的房子我们不能偷,也就是说,我们的第k-1个房子是不能偷的,但是我们可以偷第k-2个房子,也就是说在前k-2间房子的偷取金额加上当前的偷取房子,就是这种可能性的金额;第二个,如果我们选择不偷的话,那么,旁边的房子我们就可以考虑了,也就是前k-1个房子偷取的金额数。

这样,我们的转移方程其实就分析出来了:

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

注意,我们需要从i=2开始遍历,因为0,1我们在前面已经考虑过了,所以只需要从第3个开始考虑即可。

上代码:

复制代码
class Solution {
public:
    int rob(vector<int>& nums) {
        int n=nums.size();
        int sum=0;
        vector<int>dp(n,0);
        if(n==0)
        return 0;
        else if(n==1)
        return nums[0];
        dp[0]=nums[0];
        dp[1]=max(nums[0],nums[1]);
        for(int i=2;i<n;i++){
            dp[i]=max(dp[i-2]+nums[i],dp[i-1]);
        }
        return dp[n-1];
    }
};
相关推荐
ffqws_1 天前
A*算法:P5507 机关 题解
算法
执着2591 天前
力扣hot100 - 108、将有序数组转换为二叉搜索树
算法·leetcode·职场和发展
2501_901147831 天前
学习笔记:单调递增数字求解的迭代优化与工程实践
linux·服务器·笔记·学习·算法
AI科技星1 天前
张祥前统一场论核心场方程的经典验证-基于电子与质子的求导溯源及力的精确计算
线性代数·算法·机器学习·矩阵·概率论
kebijuelun1 天前
ERNIE 5.0:统一自回归多模态与弹性训练
人工智能·算法·语言模型·transformer
历程里程碑1 天前
普通数组----最大子数组和
大数据·算法·elasticsearch·搜索引擎·排序算法·哈希算法·散列表
52Hz1181 天前
力扣230.二叉搜索树中第k小的元素、199.二叉树的右视图、114.二叉树展开为链表
python·算法·leetcode
苦藤新鸡1 天前
56.组合总数
数据结构·算法·leetcode
菜鸟233号1 天前
力扣647 回文子串 java实现
java·数据结构·leetcode·动态规划