[LeetCode] 120. 三角形最小路径和 ☆☆☆(动态规划 范例)

三角形最小路径和  看看图就行

描述

给定一个三角形,找出自顶向下的最小路径和。每一步只能移动到下一行中相邻的结点上。

例如,给定三角形:

[
[2],
[3,4],
[6,5,7],
[4,1,8,3]
]
自顶向下的最小路径和为 11(即,2 + 3 + 5 + 1 = 11)。

说明:

如果你可以只使用 O(n) 的额外空间(n 为三角形的总行数)来解决这个问题,那么你的算法会很加分。

解析

递归

每个元素,往下的元素最多2种。所以,可以递归,一个个比较最大值出来。

递归优化(可以用动态规划理解)

从倒数第二个list开始,临时存储当前行的每个元素 + 下一行可能的2个元素的最小值。依次往上递推,最后返回的第一行即是解。

动态规划(二维数组)

参照递归优化的理解,可以推出二维数组的解。

动态规划(一维数组)

参照二维数组的解,可以得知,计算每个元素,只受2个值影响。

代码

递归(超时)

static int CUR_RES = Integer.MAX_VALUE;
    public static int minimumTotal(List<List<Integer>> triangle) {
        for (int i = 0; i < triangle.get(0).size(); i++) {
            miniHelp(triangle, 0, i, triangle.get(0).get(i));
        }
        return CUR_RES;
    }

    public static void miniHelp(List<List<Integer>> triangle, int startH, int startL, int res) {
        if (startH == triangle.size() - 1) {
            if (res < CUR_RES) {
                CUR_RES = res;
            }
            return;
        }
        List<Integer> nextList = triangle.get(startH + 1);
        miniHelp(triangle, startH + 1, startL, res + nextList.get(startL));
        miniHelp(triangle, startH + 1, startL + 1, res + nextList.get(startL + 1));
    }

递归优化(超时)

public static int minimumTotal2(List<List<Integer>> triangle) {
        return minimumTotal2H(triangle, 0, 0);
    }

    public static int minimumTotal2H(List<List<Integer>> triangle, int listIndex, int col) {
        if (listIndex == triangle.size() - 1) {
            return triangle.get(listIndex).get(col);
        }
        int left = minimumTotal2H(triangle, listIndex + 1, col);
        int right = minimumTotal2H(triangle, listIndex + 1, col + 1);
        int curVal = triangle.get(listIndex).get(col);
        int res = Math.min(left, right) + curVal;
        return res;
    }

动态规划(二维数组)

public static int minimumTotal5(List<List<Integer>> triangle) {
        int length = triangle.size();
        int[][] dp = new int[length][length];
        List<Integer> lastList = triangle.get(length - 1);
        for (int i = 0; i < lastList.size(); i++) {
            dp[length - 1][i] = lastList.get(i);//最后一行初始值,其实也可以放在下面的循环。放这里清晰点
        }
        for (int i = length - 2; i >= 0; i--) {
            List<Integer> nowList = triangle.get(i);
            for (int k = 0; k < nowList.size(); k++) {
                dp[i][k] = Math.min(dp[i + 1][k], dp[i + 1][k + 1]) + nowList.get(k);
            }
        }
        return dp[0][0];
    }

动态规划(一维数组)

public static int minimumTotal6(List<List<Integer>> triangle) {
        int length = triangle.size();
        int[] dp = new int[length + 1];
        for (int i = length - 1; i >= 0; i--) {
            List<Integer> nowList = triangle.get(i);
            for (int k = 0; k < nowList.size(); k++) {
                dp[k] = Math.min(dp[k], dp[k + 1]) + nowList.get(k);
            }
        }
        return dp[0];
    }

猜你喜欢

转载自www.cnblogs.com/fanguangdexiaoyuer/p/12162986.html