Bootstrap

代码随想录Day34 | 62.不同路径,63.不同路径II,343.整数拆分,96.不同的二叉搜索树

代码随想录Day34 | 62.不同路径,63.不同路径II,343.整数拆分,96.不同的二叉搜索树

62.不同路径

动态规划第二集:

比较标准简单的一道动态规划,状态转移方程容易想到

难点在于空间复杂度的优化,详见代码

class Solution {
    public int uniquePaths(int m, int n) {
        // 标准的动态规划
        int[][] dp = new int[m + 1][n + 1];
        // 初始化时多加了一行一列,方便初始化
        dp[1][0] = 1;
        for (int i = 1; i < dp.length; i++) {
            for (int j = 1; j < dp[0].length; j++) {
                // 状态转移方程
                dp[i][j] = dp[i][j - 1] + dp[i - 1][j];
            }
        }
        return dp[m][n];
    }
}

class Solution {
    public int uniquePaths(int m, int n) {
        // 标准的动态规划,空间优化版
        int[] dp = new int[n + 1];
        dp[1] = 1;
        for (int i = 1; i <= m; i++) {
            for (int j = 2; j <= n; j++) {
                // 状态转移方程
                // 只需要第 i 行与第 i-1 行的数据
                // dp[j - 1]已更新,是第 i 行的数据
                // dp[j]未更新,是第 i-1 行的数据
                dp[j] = dp[j - 1] + dp[j];
            }
        }
        return dp[n];
    }
}

63.不同路径II

相比上题只多了一个障碍的判断

class Solution {
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int m = obstacleGrid.length;
        int n = obstacleGrid[0].length;
        // 空间优化思路同62题
        int[] dp = new int[n + 1];
        dp[1] = 1;
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                // 处理障碍情况
                if (obstacleGrid[i - 1][j - 1] == 1)
                    dp[j] = 0;
                // 状态转移方程
                else dp[j] = dp[j - 1] + dp[j];
            }
        }
        return dp[n];
    }
}

343.整数拆分

动态规划问题,相对简单,想清楚状态转移方程就好,详见代码注释

class Solution {
    public int integerBreak(int n) {
        // dp[i] 的定义是 对 i 进行划分后的最大乘积
        int[] dp = new int[n + 1];
        dp[2] = 1;
        // 动态规划
        for (int i = 3; i <= n; i++) {
            // 循环进行划分
            for (int j = 1; j <= i / 2; j++) {
                // 状态转移方程
                // j * dp[i - j] 相当于是 在 i-j 中进行了多次划分
                // j * (i - j) 是只划分一次
                dp[i] = Math.max(dp[i], Math.max(j * dp[i - j], j * (i - j)));
            }
        }
        return dp[n];
    }
}

96.不同的二叉搜索树

动态规划:

要注意到,二叉树种类数目 = 左子树种类数目 * 右子树种类数目

class Solution {
    public int numTrees(int n) {
        // dp[i]定义为 i个节点时,互不相同的BST的种类数
        int[] dp = new int[n + 1];
        // 初始化:0个节点时只有一种
        dp[0] = 1;
        for (int i = 1; i <= n ; i++) {
            // 循环选择根节点为 j
            for (int j = 1; j <= i; j++) {
                // dp[j - 1]为左子树种类数,dp[i - j]为右子树种类数
                // 左右数目相乘即为根节点为 j 时的种类数
                // 累加到 dp[i] 上
                dp[i] += dp[j - 1] * dp[i - j];
            }
        }
        return dp[n];
    }
}
;