力扣第39天----第198题、第213题、第337题

力扣第39天----第198题、第213题、第337题

文章目录

一、第198题--打家劫舍

​ 经过前面背包题目的训练,这题挺简单,没啥太多可说的。递推公式,考虑当前点是否采用,分采用、不采用2种思路进行。

c++ 复制代码
class Solution {
public:
    int rob(vector<int>& nums) {
        vector<int> dp(nums.size(), 0);
        dp[0] = nums[0];
        if (nums.size() <2) return nums[0];
        dp[1] = max(nums[0], nums[1]);
        for (int i = 2; i< nums.size(); ++i){
            dp[i] = max(dp[i-1], dp[i-2] + nums[i]);
        }
        return dp[nums.size() - 1];
    }
};

二、第213题--打家劫舍II

​ 环形问题,展开成线性问题,展开成2种线性结构,再分别考虑。

c++ 复制代码
class Solution {
public:
    int rob_linear(vector<int>& nums) {
        vector<int> dp(nums.size(), 0);
        dp[0] = nums[0];
        if (nums.size() <2) return nums[0];
        dp[1] = max(nums[0], nums[1]);
        for (int i = 2; i< nums.size(); ++i){
            dp[i] = max(dp[i-1], dp[i-2] + nums[i]);
        }
        return dp[nums.size() - 1];
    }

    int rob(vector<int>& nums) {
        if (nums.size() == 1) return nums[0];
        vector<int> nums10(nums.begin(), nums.end()-1);
        int line10 = rob_linear(nums10);

        vector<int> nums01(nums.begin()+1, nums.end());
        int line01 = rob_linear(nums01);
        return max(line01, line10);
    }
};

三、第337题--打家劫舍 III

​ 树形dp----没基础的话,不太好做;跟着做下来,基本消化了吧。

​ dp数组----每个节点不采用、采用状态下,得到的金钱;所以,就是个二维数组。

​ 遍历顺序----采用树的遍历方式,这里采用后序遍历。子节点处理完了,才能处理根节点。

​ 初始化----这里与截止条件重合,树遍历终止的时候,就是dp数组初始化的时候,初始化为{0,0},即空节点时价值为0。

​ 递推公式----分采用当前节点、不采用当前节点2种情况分别递推。采用的时候,下一层就不能采用,就是left[0]、right[0],然后再加起来;不采用的时候,下一层就要综合考虑,所以max(left[0], left[1])、max(right[0], right[1]),左右分支都要考虑到,再加一起。

c++ 复制代码
class Solution {
public:
    int rob(TreeNode* root) {
        vector<int> result = robTree(root);
        return max(result[0], result[1]);
    }
    vector<int> robTree(TreeNode* root){    //0表示不采取当前点,1表示采取当前点。
        if (root == nullptr) return vector<int> {0,0};
        vector<int> left = robTree(root->left);
        vector<int> right = robTree(root->right);

        int cur_get = root->val + left[0] + right[0];
        int cur_noget = max(left[0], left[1]) + max(right[0], right[1]);
        return vector<int> {cur_noget, cur_get};
    }
};
相关推荐
夜思红尘4 小时前
算法--双指针
python·算法·剪枝
散峰而望5 小时前
【算法竞赛】C++函数详解:从定义、调用到高级用法
c语言·开发语言·数据结构·c++·算法·github
CoderCodingNo5 小时前
【GESP】C++五级真题(贪心思想考点) luogu-B4071 [GESP202412 五级] 武器强化
开发语言·c++·算法
我有一些感想……5 小时前
An abstract way to solve Luogu P1001
c++·算法·ai·洛谷·mlp
前端小L5 小时前
双指针专题(三):去重的艺术——「三数之和」
javascript·算法·双指针与滑动窗口
在风中的意志5 小时前
[数据库SQL] [leetcode] 2388. 将表中的空值更改为前一个值
数据库·sql·leetcode
智者知已应修善业6 小时前
【求等差数列个数/无序获取最大最小次大次小】2024-3-8
c语言·c++·经验分享·笔记·算法
还不秃顶的计科生6 小时前
LeetCode 热题 100第二题:字母易位词分组python版本
linux·python·leetcode
LYFlied6 小时前
【每日算法】LeetCode 416. 分割等和子集(动态规划)
数据结构·算法·leetcode·职场和发展·动态规划
多米Domi0117 小时前
0x3f 第19天 javase黑马81-87 ,三更1-23 hot100子串
python·算法·leetcode·散列表