[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 };