hot100打卡——day08

337. 打家劫舍 III

https://leetcode.cn/problems/house-robber-iii/description/?envType=problem-list-v2&envId=2cktkvj

java 复制代码
class Solution {
    public int rob(TreeNode root) {
        int[] dp = dfs(root);
        return Math.max(dp[0], dp[1]);
    }

    // dp[0] 表示当前节点取  dp[1] 表示当前节点不取
    int[] dfs(TreeNode node) {
        if (node == null) return new int[] {0, 0};

        int[] left = dfs(node.left);
        int[] right = dfs(node.right);

        // 取当前节点
        int cur = node.val + left[1] + right[1];
        // 不取当前节点
        int notCur = Math.max(left[0], left[1]) + Math.max(right[0], right[1]);
        
        return new int[] {cur, notCur};
        
    }
}

这道题直接使用树形dp进行解决就行了,每次都是返回要取当前节点和不取当前节点两种情况

121. 买卖股票的最佳时机

https://leetcode.cn/problems/best-time-to-buy-and-sell-stock/description/?envType=problem-list-v2&envId=2cktkvj

java 复制代码
class Solution {
    public int maxProfit(int[] prices) {
        int n = prices.length;
        // dp[i][0] 表示第i天持有股票的最大盈利
        // dp[i][1] 表示第i天不持有股票的最大盈利
        int[][] dp = new int[n][2]; 
        dp[0][0] = -prices[0];
        dp[0][1] = 0;
        for (int i = 1; i < n; i++) {
            dp[i][0] = Math.max(dp[i-1][0], dp[i][1] - prices[i]);
            dp[i][1] = Math.max(dp[i-1][1], dp[i][0] + prices[i]);
        }
        return dp[n-1][1];
    }
}

每一天的状态无非就是持有股票和不持有两种状态,那我们要做的就是分别统计持有和不持有的一个最大盈利情况

注意:第i天不持有,可能是第i-1天有,第i天卖出,也可能是从第i-1天就没有;第i天有,可能是从第i-1天就有,也可能是第i-1天没有,第i天刚买入的

309. 买卖股票的最佳时机含冷冻期

https://leetcode.cn/problems/best-time-to-buy-and-sell-stock-with-cooldown/?envType=problem-list-v2&envId=2cktkvj

java 复制代码
class Solution {
    public int maxProfit(int[] prices) {
        int n = prices.length;

        if (n == 1) return 0;
        if (n == 2) return Math.max(prices[1]-prices[0], 0);

        // dp[i][0] 表示第i天不持有
        // dp[i][1] 表示第i天持有
        int[][] dp = new int[n + 1][2];

        dp[0][1] = Integer.MIN_VALUE; // 不可能
        dp[1][1] = -prices[0];
        dp[0][0] = 0;
        dp[1][0] = 0;

        for (int i = 1; i < n; i++) {
            dp[i+1][0] = Math.max(dp[i][0], dp[i][1] + prices[i]);
            dp[i+1][1] = Math.max(dp[i][1], dp[i-1][0] - prices[i]);
        }
        return dp[n][0];
    }
}

312. 戳气球

https://leetcode.cn/problems/burst-balloons/description/?envType=problem-list-v2&envId=2cktkvj

java 复制代码
class Solution {
     public int maxCoins(int[] nums) {
        int n = nums.length + 2;
        int[] bo = new int[n];
        bo[0] = bo[n-1] = 1;

        for (int i = 1; i < n-1; i++) bo[i] = nums[i-1];
        
        int[][] dp = new int[n][n];

        for (int i = n - 3; i >= 0; i--) {
            for (int j = i+2; j < n; j++) {
                for (int k = i+1; k < j; k++) {
                    int sum = bo[i] * bo[k] * bo[j] + dp[i][k] + dp[k][j];
                    dp[i][j] = Math.max(dp[i][j], sum);
                }
            }
        }
        return dp[0][n-1];
    }
}

这道题的解题思路就是倒着来,我们首先在原来的数组两边分别塞入一个1,这样对边界的处理就会更加的简单

我们可以将问题进行拆解,首先我们需要不断的枚举区间,当然区间的大小至少是3,然后我们需要枚举区间中每一个元素被戳破时的最大价值,也就是bo[i] * bo[k] * bo[j] + dp[i][k] + dp[k][j]

相关推荐
a努力。2 小时前
国家电网Java面试被问:慢查询的优化方案
java·开发语言·面试
l1t2 小时前
DeepSeek总结的算法 X 与舞蹈链文章
前端·javascript·算法
gihigo19982 小时前
水声信号处理中DEMON谱分析的原理、实现与改进
算法·信号处理
歌_顿2 小时前
微调方法学习总结(万字长文!)
算法
@小码农2 小时前
202512 电子学会 Scratch图形化编程等级考试四级真题(附答案)
java·开发语言·算法
程序猿ZhangSir2 小时前
深入理解 BIO,NIO,AIO 三者的用途和区别?Select,poll,epoll 操作系统函数简介
java·spring·nio
智航GIS2 小时前
6.2 while循环
java·前端·python
2201_757830872 小时前
AOP核心概念
java·前端·数据库
为所欲为、Lynn2 小时前
用FastJson的Filter自动映射枚举
java·spring boot