leetcode-打家劫舍

题目:打家劫舍I

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。
给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

在这里插入图片描述

解题思路

首先考虑最简单的情况。如果只有一间房屋,则偷窃该房屋,可以偷窃到最高总金额。如果只有两间房屋,则由于两间房屋相邻,不能同时偷窃,只能偷窃其中的一间房屋,因此选择其中金额较高的房屋进行偷窃,可以偷窃到最高总金额。

如果房屋数量大于两间,应该如何计算能够偷窃到的最高总金额呢?对于第k (k>2) 间房屋,有两个选项:

偷窃第k 间房屋,那么就不能偷窃第 k−1 间房屋,偷窃总金额为前 k−2 间房屋的最高总金额与第 k间房屋的金额之和。

不偷窃第 k 间房屋,偷窃总金额为前 k−1 间房屋的最高总金额。

在两个选项中选择偷窃总金额较大的选项,该选项对应的偷窃总金额即为前 k 间房屋能偷窃到的最高总金额。
dp[i] 表示前 i 间房屋能偷窃到的最高总金额,那么就有如下的状态转移方程:
在这里插入图片描述
边界条件为:
在这里插入图片描述
最终的答案即为dp[n−1],其中 n 是数组的长度。

C++实现

class Solution {
public:
    int rob(vector<int>& nums) {
        int n=(int)nums.size();
        if(n==1)
            return nums[0];
        //空间复杂度降低到常数级
        int first=nums[0],second=max(nums[0],nums[1]);
        for(int i=2;i<n;++i)
        {
            int tmp=second;
            second=max(first+nums[i],second);
            first=tmp;
        }
	return second;
    }
};

  1. 时间复杂度:O(n)
  2. 空间复杂度:O(1)

题目:打家劫舍II

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警 。
给定一个代表每个房屋存放金额的非负整数数组,计算你 在不触动警报装置的情况下 ,今晚能够偷窃到的最高金额。

在这里插入图片描述

解题思路

在打家劫舍I的基础上考虑这样一个问题:
**如何才能保证第一间房屋和最后一间房屋不同时偷窃呢?**如果偷窃了第一间房屋,则不能偷窃最后一间房屋,因此偷窃房屋的范围是第一间房屋到最后第二间房屋;如果偷窃了最后一间房屋,则不能偷窃第一间房屋,因此偷窃房屋的范围是第二间房屋到最后一间房屋。
假设数组nums 的长度为 nn。如果不偷窃最后一间房屋,则偷窃房屋的下标范围是[0,n−2];如果不偷窃第一间房屋,则偷窃房屋的下标范围是 [1,n−1]。在确定偷窃房屋的下标范围之后,即可用第 198 题的方法解决。对于两段下标范围分别计算可以偷窃到的最高总金额,其中的最大值即为在 n 间房屋中可以偷窃到的最高总金额。

假设偷窃房屋的下标范围是[start,end],用 dp[i] 表示在下标范围[start,i] 内可以偷窃到的最高总金额,那么就有如下的状态转移方程:
在这里插入图片描述
边界条件为:
在这里插入图片描述
分别取 (start,end)=(0,n−2) 和 (start,end)=(1,n−1) 进行计算,取两个dp[end] 中的最大值,即可得到最终结果。

C++实现

class Solution {
public:
    int rob(vector<int>& nums) {
        int n=(int)nums.size();
        if(n==1)
            return nums[0];
        else if(n==2)
            return max(nums[0],nums[1]);
        else
            return max(Subrob(nums,0,n-2),Subrob(nums,1,n-1));
    }
    int Subrob(vector<int>& nums,int start,int end) {
        int n=end-start+1;
        if(n==1)
            return nums[start];
        int first=nums[start],second=max(nums[start],nums[start+1]);
        for(int i=start+2;i<=end;++i)
        {
            int tmp=second;
            second=max(first+nums[i],second);
            first=tmp;
        }
        return second;
    }
};
  1. 时间复杂度:O(n)
  2. 空间复杂度:O(1)

337-打家劫舍叁

小偷又发现了一个新的可行窃的地区。这个地区只有一个入口,我们称之为 root 。

除了 root 之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果 两个直接相连的房子在同一天晚上被打劫 ,房屋将自动报警。

给定二叉树的 root 。返回 在不触动警报的情况下 ,小偷能够盗取的最高金额 。
在这里插入图片描述
在这里插入图片描述

动态规划

化一下这个问题:一棵二叉树,树上的每个点都有对应的权值,每个点有两种状态(选中和不选中),问在不能同时选中有父子关系的点的情况下,能选中的点的最大权值和是多少。

我们可以用 f(o) 表示选择 o 节点的情况下,o 节点的子树上被选择的节点的最大权值和;g(o) 表示不选择 o节点的情况下,o 节点的子树上被选择的节点的最大权值和;l和 r 代表 o 的左右孩子。

当 o 被选中时,o的左右孩子都不能被选中,故 o 被选中情况下子树上被选中点的最大权值和为 l和 r 不被选中的最大权值和相加,即 f(o)=g(l)+g(r)。
当 o 不被选中时,o 的左右孩子可以被选中,也可以不被选中。对于 o 的某个具体的孩子 x,它对 o 的贡献是 x 被选中和不被选中情况下权值和的较大值。故g(o)=max{f(l),g(l)}+max{f(r),g(r)}。

至此,我们可以用哈希表来存 f 和 g 的函数值,用深度优先搜索的办法后序遍历这棵二叉树,我们就可以得到每一个节点的 f 和 g。根节点的 f和 g 的最大值就是我们要找的答案。

我们不难给出这样的实现:

class Solution {
public:
    unordered_map <TreeNode*, int> f, g;

    void dfs(TreeNode* node) {
        if (!node) {
            return;
        }
        dfs(node->left);
        dfs(node->right);
        //f[NULL]=G[NULL]=0
        f[node] = node->val + g[node->left] + g[node->right];
        g[node] = max(f[node->left], g[node->left]) + max(f[node->right], g[node->right]);
    }

    int rob(TreeNode* root) {
        dfs(root);
        return max(f[root], g[root]);
    }
};

假设二叉树的节点个数为 n。

我们可以看出,以上的算法对二叉树做了一次后序遍历,时间复杂度是 O(n);由于递归会使用到栈空间,空间代价是 O(n),哈希表的空间代价也是 O(n),故空间复杂度也是O(n)。

我们可以做一个小小的优化,我们发现无论是 f(o) 还是 g(o),他们最终的值只和 f(l)、g(l)、f®、g® 有关,所以对于每个节点,我们只关心它的孩子节点们的 f和 g 是多少。我们可以设计一个结构,表示某个节点的 f 和 g 值,在每次递归返回的时候,都把这个点对应的 f 和 g 返回给上一级调用,这样可以省去哈希表的空间。

代码如下。

//结构体返回值得顺序是变量声明的顺序
struct SubtreeStatus {
    int selected;
    int notSelected; 
};

class Solution {
public:
    SubtreeStatus dfs(TreeNode* node) {
        if (!node) {
            return {0, 0};
        }
        auto l = dfs(node->left);
        auto r = dfs(node->right);
        int selected = node->val + l.notSelected + r.notSelected;
        int notSelected = max(l.selected, l.notSelected) + max(r.selected, r.notSelected);
        return {selected, notSelected};
    }

    int rob(TreeNode* root) {
        auto rootStatus = dfs(root);
        return max(rootStatus.selected, rootStatus.notSelected);
    }
};
  1. 时间复杂度:O(N)
  2. 空间复杂度:O(N)
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值