zoukankan      html  css  js  c++  java
  • [Leetcode] 第337题 打家劫舍III

    一、题目描述

    在上次打劫完一条街道之后和一圈房屋后,小偷又发现了一个新的可行窃的地区。这个地区只有一个入口,我们称之为“根”。 除了“根”之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果两个直接相连的房子在同一天晚上被打劫,房屋将自动报警。

    计算在不触动警报的情况下,小偷一晚能够盗取的最高金额。

    示例 1:

    输入: [3,2,3,null,3,null,1]
    
         3
        / 
       2   3
            
         3   1
    
    输出: 7 
    解释: 小偷一晚能够盗取的最高金额 = 3 + 3 + 1 = 7.

    示例 2:

    输入: [3,4,5,1,3,null,1]
    
         3
        / 
       4   5
      /     
     1   3   1
    
    输出: 9
    解释: 小偷一晚能够盗取的最高金额 = 4 + 5 = 9.

    二、题目分析和代码实现

    二叉树的题目首先就应该想到深度优先搜索,先解决子树,再考虑当前树

    1、思路一,标记父节点的状态

    1)用flag来标记父节点是不是已经被盗取了
    2)如果没有被盗取,那么可以选择盗取当前的节点,也可以选择不盗取,取两者比较大的。
    3)如果父节点被盗取了,那么当前的节点一定不能被盗取

     1 class Solution {
     2 public:
     3     int rob(TreeNode* root) {
     4         return dfs(root, false);
     5     }
     6 private:
     7     int dfs(TreeNode *t, bool flag) {
     8         if (!t)return 0;
     9         int tmp = dfs(t->left, false) + dfs(t->right, false);
    10         if (flag)
    11             return tmp;
    12         else return max(t->val + dfs(t->left, true) + dfs(t->right, true), tmp);
    13     }
    14 };

    我们可以看到,代码中大量有重复计算的部分,先计算了不包含父节点的情况,又计算了包含父节点的情况,这样导致时间复杂度比较高

    2、接下来第二个方法避免了上述问题

    1)有一个改进的方法是用含有2个元素的数组分别代表含有当前节点的值,和不含有当前节点的值
    2)直接返回数组,然后在最后让根节点进行选择

     1 class Solution {
     2 public:
     3     int rob(TreeNode* root) {
     4         vector<int>res = dfs(root);
     5         return max(res[0], res[1]);
     6     }
     7 private:
     8     vector<int> dfs(TreeNode *t) {
     9         vector<int>res(2);
    10         if (!t) {
    11             res[0] = 0;
    12             res[1] = 0;
    13             return res;
    14         }
    15         vector<int> left_val = dfs(t->left);
    16         vector<int> right_val = dfs(t->right);
    17         res[0] = t->val + left_val[1] + right_val[1];//包含当前节点,就不能包含左右子节点
    18         res[1] = max(left_val[0], left_val[1]) + max(right_val[0], right_val[1]);//不包含当前节点,可以包含也可以不包含子节点,选择较大的那个
    19         return res;
    20     }
    21 };
  • 相关阅读:
    C#中的const和readonly之间的不同(转)
    文字在状态栏上从右往左显示,而且是循环的
    文字在状态栏上从左往右一个一个地显示
    猛然发现,已经第100篇随笔了
    怎样使按钮响应回车键
    编程之我见(二 类库)初露牛角
    编程之我见(一 语言)小试牛刀
    开始→运行→输入的命令集锦(转)收藏
    显示走动的数字时间和显示星期,年,月,日
    在两个页面之间互相写其控件内的值
  • 原文地址:https://www.cnblogs.com/zhizhiyu/p/10221064.html
Copyright © 2011-2022 走看看