跳到主要内容

15、算法与数据结构 - 实战:动态规划

什么是动态规划

已经走过的分支,用一张表记着,下次遇到相同的分支,直接从表中拿值,不走重复的支路。

比如斐波那契数列,求f(n),则要计算f(n-1),f(n-2),而计算f(n-1)时,则要计算f(n-2),f(n-3),当计算完f(n-1),回到f(n)的计算,去计算f(n-2),其实前面已经算过,如果用一张表记着,所以这里就不用重复算。

 

1、假设有排成一行的N个位置,记为1~N,N 一定大于或等于 2,开始时机器人在其中的M位置上(M 一定是 1~N 中的一个)。如果机器人来到1位置,那么下一步只能往右来到2位置;如果机器人来到N位置,那么下一步只能往左来到 N-1 位置;如果机器人来到中间位置,那么下一步可以往左走或者往右走;规定机器人必须走 K 步,最终能来到P位置(P也是1~N中的一个)的方法有多少种?

/**
 * 假设有排成一行的N个位置,记为1~N,N 一定大于或等于 2,
 * 开始时机器人在其中的M位置上(M 一定是 1~N 中的一个)。
 * 如果机器人来到1位置,那么下一步只能往右来到2位置;
 * 如果机器人来到N位置,那么下一步只能往左来到 N-1 位置;
 * 如果机器人来到中间位置,那么下一步可以往左走或者往右走;
 * 规定机器人必须走 K 步,最终能来到P位置(P也是1~N中的一个)的方法有多少种?
 * 给定四个参数 N、M、K、P,返回方法数。
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming01 {
   
     

    /**
     * 暴力递归求解
     * @param n 总位置数
     * @param m 初始化位置
     * @param k 可走步数
     * @param p 目标位置
     * @return
     */
    public static int method01(int n, int m, int k, int p) {
   
     
        // 不合法参数判断
        if (n < 2 || k < 1 || m < 1 || m > n || p < 1 || p > n) return 0;
        return processo1(n, m, k, p);
    }

    /**
     * 暴力递归求解
     * @param n 总位置数
     * @param curr 当前位置
     * @param remain 剩余可走步数
     * @param p 目标位置
     * @return
     */
    private static int processo1(int n, int curr, int remain, int p) {
   
     
        // base case:还剩0步要走,如果当前位置curr在目标位置p,则发现一种方法
        if (remain == 0) return curr == p ? 1 : 0;
        // 1位置,只能往右走
        if (curr == 1) return processo1(n, 2, remain - 1, p);
        // n位置,只能往左走
        if (curr == n) return processo1(n, n - 1, remain - 1, p);
        // 两个递归 往左走,往右走 返回的方法数累加
        return processo1(n, n - 1, remain - 1, p) + processo1(n, n + 1, remain - 1, p);
    }

    /**
     * 粗糙的动态规划,记忆化搜索
     * @param n 总位置数
     * @param m 初始化位置
     * @param k 可走步数
     * @param p 目标位置
     * @return
     */
    public static int method02(int n, int m, int k, int p) {
   
     
        // 不合法参数判断
        if (n < 2 || k < 1 || m < 1 || m > n || p < 1 || p > n) return 0;
        // 初始化缓存表
        int[][] dp = new int[n + 1][k + 1];
        for (int i = 0; i <= n; i++) {
   
     
            for (int j = 0; j <= k; k++) {
   
     
                dp[i][j] = -1;
            }
        }
        return processo2(n, m, k, p, dp);
    }

    /**
     * 粗糙的动态规划,记忆化搜索
     * @param n 总位置数
     * @param curr 当前位置
     * @param remain 剩余可走步数
     * @param p 目标位置
     * @param dp 记忆表
     * @return
     */
    private static int processo2(int n, int curr, int remain, int p, int[][] dp) {
   
     
        // 如果缓存中有值,直接拿值,返回
        if (dp[curr][remain] != -1) return dp[curr][remain];
        if (remain == 0) {
   
     
            dp[curr][remain] = curr == p ? 1 : 0;
            return dp[curr][remain];
        }
        if (curr == 1) {
   
     
            // 返回得到的值前,先存缓存表
            dp[curr][remain] = processo2(n, 2, remain - 1, p, dp);
            return dp[curr][remain];
        }
        if (curr == n) {
   
     
            // 返回得到的值前,先存缓存表
            dp[curr][remain] = processo2(n, n - 1, remain - 1, p, dp);
            return dp[curr][remain];
        }
        // 返回得到的值前,先存缓存表
        dp[curr][remain] = processo2(n, curr - 1, remain - 1, p, dp) + processo2(n, curr + 1, remain - 1, p, dp);
        return dp[curr][remain];
    }

    /**
     * 动态规划求解
     * @param n 总位置数
     * @param m 初始化位置
     * @param k 可走步数
     * @param p 目标位置
     * @return
     */
    public static int method03(int n, int m, int k, int p) {
   
     
        // 不合法参数判断
        if (n < 2 || k < 1 || m < 1 || m > n || p < 1 || p > n) return 0;
        /*
          dp表
          dp[i][j]
          表示当前在i位置
          表示还剩j步要走
         */
        int[][] dp = new int[n + 1][k + 1];
        /*for (int i = 0; i <= n; i++) {

            dp[i][0] = 0;
        }
        for (int j = 0; j <= k; k++) {
            dp[0][j] = 0;
        }*/
        // 根据暴力递归的base case,初始化
        dp[p][0] = 1;
        /*
        根据暴力递归的依赖关系,填表
        dp[1][j]依赖dp[2][j-1]
        dp[n][j]依赖dp[n-1][j-1]
        dp[i][j]依赖dp[i-1][j-1]和dp[i+1][j-1]
         */
        for (int remain = 1; remain <= k; remain++) {
   
     
            for (int curr = 1; curr <= n; curr++) {
   
     
                if (curr == 1) {
   
     
                    dp[curr][remain] = dp[2][remain - 1];
                } else if (curr == n) {
   
     
                    dp[curr][remain] = dp[n - 1][remain - 1];
                } else {
   
     
                    dp[curr][remain] = dp[curr + 1][remain - 1] + dp[curr - 1][remain - 1];
                }
            }
        }
        return dp[m][k];
    }

}

2、背包问题(动态规划求解)

/**
 * 背包问题(动态规划求解)
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming02 {
   
     

    public static int processByRecursive(int[] weights, int[] values, int bag) {
   
     
        return byRecursive(weights, values, 0, bag);
    }

    /**
     * 暴力递归
     * @param weights 物品总量
     * @param values 物品价值
     * @param index 当前物品下标
     * @param space 背包剩余空间
     * @return
     */
    private static int byRecursive(int[] weights, int[] values, int index, int space) {
   
     
        if (space < 0 || index == weights.length) return 0;
        // 当前位置物品不要,直接下一轮递归
        int p1 = byRecursive(weights, values, index + 1, space);
        int p2 = Integer.MIN_VALUE;
        if (space >= weights[index]) {
   
     
            //如果背包还有空间放得下当前物品,参数装入当前物品,进行下一轮递归
            p2 = values[index] + byRecursive(weights, values, index + 1, space - weights[index]);
        }
        //递归返回,总p1,p2中取最大值
        return Math.max(p1, p2);
    }

    /**
     * 动态规划
     * @param weights 物品总量
     * @param values 物品价值
     * @param bag 背包能承载的重量
     * @return
     */
    public static int processByDynamicProgramming(int[] weights, int[] values, int bag) {
   
     
        int[][] dp = new int[values.length + 1][bag + 1];
        for (int index = values.length - 1; index >= 0; index--) {
   
     
            for (int remain = 0; remain <= bag; remain--) {
   
     
                // 当前位置物品不要,直接下一轮递归
                int p1 = dp[index + 1][remain];
                int p2 = Integer.MIN_VALUE;
                if (remain >= weights[index]) {
   
     
                    //如果背包还有空间放得下当前物品,参数装入当前物品,进行下一轮递归
                    p2 = values[index] + dp[index + 1][remain - weights[index]];
                }
                dp[index][remain] = Math.max(p1, p2);
            }
        }
        return dp[0][bag];
    }

}

3、给定一个只由数字组成的字符串,1对应A,2对应B…,将字符串的数据转化为字母,如11,可以转化为AA,也可以转化为K,问:有多少种转化结果

/**
 * 给定一个只由数字组成的字符串,1对应A,2对应B...
 * 将字符串的数据转化为字母,如11,可以转化为AA,也可以转化为K
 * 问:有多少种转化结果
 */
public class DynamicProgramming03 {
   
     

    public static int getThransferCountByRecursive(String str) {
   
     
        char[] chars = str.toCharArray();
        return process(chars, 0);
    }

    /**
     * 暴力递归
     * @param chars 数字字符数组
     * @param i 当前位置
     * @return
     */
    private static int process(char[] chars, int i) {
   
     
        if (i == chars.length) return 1;
        int res = 0;
        // 如当前位置字符是0,代表之前做的决定有问题
        if (chars[i] == '0') return res;
        res += process(chars, i + 1);
        //当前数字位1,则有两个分支:1.只把当前数字转换为字母,2.把当前数字和下一位合在一起转换为字母
        if (chars[i] == '1') {
   
     
            if (i + 1 < chars.length) {
   
     
                res += process(chars, i + 2);
            }
        }
        //当前数字位2,只有在下一位数字是0~6之间,才有两个分支
        if (chars[i] == '2') {
   
     
            if (i + 1 < chars.length && chars[i + 1] >='0' && chars[i + 1] <= '6') {
   
     
                res += process(chars, i + 2);
            }
        }
        //返回累加结果
        return res;
    }

    /**
     * 动态规划
     * @param str
     * @return
     */
    private static int getThransferCountByDp(String str) {
   
     
        char[] chars = str.toCharArray();
        // 因为暴力递归只有一个变化参数,所以改成动态规划后就以一位dp表
        int[] dp = new int[chars.length + 1];
        dp[chars.length] = 1;
        for (int i = chars.length - 1; i >= 0; i--) {
   
     
            int res = 0;

            // 如当前位置字符是0,代表之前做的决定有问题
            if (chars[i] == '0') {
   
     
                dp[i] = 0;
                continue;
            }

            //当前数字位1,则有两个分支:1.只把当前数字转换为字母,2.把当前数字和下一位合在一起转换为字母
            res += dp[i + 1];
            if (chars[i] == '1') {
   
     
                if (i + 1 < chars.length) {
   
     
                    res += dp[i + 2];
                }
            }

            //当前数字位2,只有在下一位数字是0~6之间,才有两个分支
            if (chars[i] == '2') {
   
     
                if (i + 1 < chars.length && chars[i + 1] >='0' && chars[i + 1] <= '6') {
   
     
                    res += dp[i + 2];
                }
            }
            
            dp[i] = res;
        }
        return dp[0];
    }

    public static void main(String[] args) {
   
     
        System.out.println(getThransferCountByRecursive("11111"));
        System.out.println(getThransferCountByDp("11111"));
    }

}

4、给定一个整形数组,代表一串纸牌,有A、B两个玩家,从中抽取纸牌,但是每次只能拿最左或者最右,假设两个玩家绝顶聪明,求最后获胜方的结果值

比如有一串纸牌50、100、20、10
因为两个玩家都绝顶聪明
先手玩家如果拿了50,后手玩家就能拿100,那么他就输了,所以他不会拿50,他会先拿10
然后剩下50、100、20
然后后手玩家看,无论拿最左还是最右的牌,他都会输,所以他只能去拿较大的牌,
然后先手玩家拿走100
所以先手玩家是必赢的
因此返回获胜方是先手玩家

假设这一串纸牌非常长,返回最后获胜的是谁

/**
 * 给定一个整形数组,代表一串纸牌
 * 有A、B两个玩家,从中抽取纸牌,但是每次只能拿最左或者最右
 * 假设两个玩家绝顶聪明(博弈)
 * 求最后获胜方的结果值
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming04 {
   
     

    /**
     * 先手拿牌
     * @param values
     * @param left
     * @param right
     * @return
     */
    public static int first(int[] values, int left, int right) {
   
     
        // 只剩下一张牌,先手拿走
        if (left == right) return values[left];
        // 因为先手决定聪明,会拿走最大的分数
        return Math.max(
                // 拿走最左侧牌,然后先手就转为后手了,调后手函数,获得剩下的分数
                values[left] + second(values, left + 1, right),
                // 拿走最右侧牌,然后先手就转为后手了,调后手函数,获得剩下的分数
                values[right] + second(values, left, right - 1)
        );
    }

    /**
     * 后手拿牌
     * @param values
     * @param left
     * @param right
     * @return
     */
    public static int second(int[] values, int left, int right) {
   
     
        // 只剩下一张牌,会被先手拿走,后手拿不到
        if (left == right) return 0;
        // 因为先手决定聪明,会让后手拿到最小的分数
        return Math.min(
                // 最左侧牌被先手拿走,后手转为先手,在left+1~right上调先手函数做决定
                first(values, left + 1, right),
                // 最右侧牌被先手拿走,后手转为先手,在left~right-1上调先手函数做决定
                first(values, left, right - 1)
        );
    }

    /**
     * 暴力递归
     * @param values
     * @return
     */
    public static int getWinValue(int[] values) {
   
     
        return Math.max(
                // 先手玩家获得的分数
                first(values, 0, values.length - 1),
                // 后手玩家获得的分数
                second(values, 0, values.length - 1)
        );
    }

    /**
     * 先手拿牌
     * @param values
     * @param left
     * @param right
     * @param firstMap
     *@param secondMap @return
     */
    public static int firstByCacheMap(int[] values, int left, int right, int[][] firstMap, int[][] secondMap) {
   
     
        // 如果缓存中有值,直接返回缓存中的值
        if (firstMap[left][right] != -1) return firstMap[left][right];
        int res = 0;
        // 只剩下一张牌,先手拿走
        if (left == right) res = values[left];
        // 因为先手决定聪明,会拿走最大的分数
        else res = Math.max(
                // 拿走最左侧牌,然后先手就转为后手了,调后手函数,获得剩下的分数
                values[left] + secondByCacheMap(values, left + 1, right, firstMap, secondMap),
                // 拿走最右侧牌,然后先手就转为后手了,调后手函数,获得剩下的分数
                values[right] + secondByCacheMap(values, left, right - 1, firstMap, secondMap)
        );
        // 记录缓存表
        firstMap[left][right] = res;
        return res;
    }

    /**
     * 后手拿牌
     * @param values
     * @param left
     * @param right
     * @param firstMap
     *@param secondMap @return
     */
    public static int secondByCacheMap(int[] values, int left, int right, int[][] firstMap, int[][] secondMap) {
   
     
        // 如果缓存中有值,直接返回缓存中的值
        if (secondMap[left][right] != -1) return secondMap[left][right];
        int res = 0;
        // 只剩下一张牌,会被先手拿走,后手拿不到
        if (left == right) res = 0;
        // 因为先手决定聪明,会让后手拿到最小的分数
        else res = Math.min(
                // 最左侧牌被先手拿走,后手转为先手,在left+1~right上调先手函数做决定
                firstByCacheMap(values, left + 1, right, firstMap, secondMap),
                // 最右侧牌被先手拿走,后手转为先手,在left~right-1上调先手函数做决定
                firstByCacheMap(values, left, right - 1, firstMap, secondMap)
        );
        // 记录缓存表
        secondMap[left][right] = res;
        return res;
    }

    /**
     * 暴力递归改成记忆化搜索
     * @param values
     * @return
     */
    public static int getWinValueByCacheMap(int[] values) {
   
     
        // 因为暴力递归中,有两个函数,先后,后手,所以搞两个缓存表
        int[][] firstMap = new int[values.length][values.length];
        int[][] secondMap = new int[values.length][values.length];
        // 初始化缓存表
        for (int i = 0; i < values.length; i++) {
   
     
            for (int j = 0; j < values.length; j++) {
   
     
                firstMap[i][j] = -1;
                secondMap[i][j] = -1;
            }
        }
        return Math.max(
                // 先手玩家获得的分数
                firstByCacheMap(values, 0, values.length - 1, firstMap, secondMap),
                // 后手玩家获得的分数
                secondByCacheMap(values, 0, values.length - 1, firstMap, secondMap)
        );
    }

    /**
     * 动态规划
     * @param values
     * @return
     */
    public static int getWinValueByDp(int[] values) {
   
     
        int n = values.length;
        // 因为暴力递归中,有两个函数,先后,后手,所以搞两个dp表
        int[][] first = new int[n][n];
        int[][] second = new int[n][n];

        // 根据暴力递归先手函数的base case(if (left == right) return values[left];),初始化first
        for (int i = 0; i < n; i++) {
   
     
            first[i][i] = values[i];
        }

        /*
        根据暴力递归推断dp表的依赖关系
        first[i][j]依赖second[i+1][j]和second[i][j-1]
        second[i][j]依赖first[i+1][j]和first[i][j-1]
        所以依着对角线从上往下填,从左往右填完所有对角线就ok
        因为left <= right,所以,左下半部分不需要填

        因为0对角线已经初始化了,从1对角线开始
         */
        for (int i = 1; i < n; i++) {
   
     
            int left = 0; // 行
            int right = i; // 列
            while (left < n - 1 && right < n) {
   
     
                first[left][right] = Math.max(values[left] + second[left + 1][right], values[right] + second[left][right - 1]);
                second[left][right] = Math.min(first[left + 1][right], first[left][right - 1]);
                // 依着对角线填,所以行++,列++
                left++;
                right++;
            }
        }
        return Math.max(first[0][n - 1], second[0][n - 1]);
    }

    public static void main(String[] args) {
   
     
        int[] values = {
   
     1,2,5,23,8,3,122,8};
        System.out.println(getWinValue(values));
        System.out.println(getWinValueByCacheMap(values));
        System.out.println(getWinValueByDp(values));
    }
}

5、货币组合问题

  • 给定一个整形数组arr表示一组不同面值的货币
  • 给定一个整形aim,表示需要通过不同的货币拼凑一起到总额aim
  • 从arr中取货币,每种货币都可以取若干张
  • 求有多少种组合方法
/**
 * 货币组合问题
 * 给定一个整形数组arr表示一组不同面值的货币
 * 给定一个整形aim,表示需要通过不同的货币拼凑一起到总额aim
 * 从arr中取货币,每种货币都可以取若干张
 * 求有多少种组合方法
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming05 {
   
     

    public static int method01(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        return process01(arr, 0, aim);
    }

    /**
     * 暴力递归
     * @param arr 货币数组
     * @param index 当前货币
     * @param remain 剩余要组出的总额
     * @return
     */
    private static int process01(int[] arr, int index, int remain) {
   
     
        if (index == arr.length) return remain == 0 ? 1 : 0;
        int res = 0;
        // index位置的货币,用0张、用1张、用2张......
        for (int i = 0; i * arr[index] <= remain; i++)
            res += process01(arr, index + 1, remain - i * arr[index]);
        return res;
    }

    public static int method02(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        // 定义并初始化缓存表
        int[][] dp = new int[arr.length + 1][aim + 1];
        for (int i = 0; i <= arr.length; i++) for (int j = 0; j <= aim; j++) dp[i][j] = -1;
        return process02(arr, 0, aim, dp);
    }

    /**
     * 记忆化搜索
     * @param arr 货币数组
     * @param index 当前货币
     * @param remain 剩余要组出的总额
     * @return
     */
    private static int process02(int[] arr, int index, int remain, int[][] dp) {
   
     
        // 命中缓存,返回
        if (dp[index][remain] != -1) return dp[index][remain];
        if (index == arr.length) return dp[index][remain] = remain == 0 ? 1 : 0;
        int res = 0;
        // index位置的货币,用0张、用1张、用2张......
        for (int i = 0; i * arr[index] <= remain; i++)
            res += process02(arr, index + 1, remain - i * arr[index], dp);
        // 记录到缓存,然后返回
        return dp[index][remain] = res;
    }

    /**
     * 动态规划
     * @param arr
     * @param aim
     * @return
     */
    public static int method03(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        /*
        dp[index][remain]
        index往后的货币任意选择,凑够remain元,有几种方法
        观察暴力递归可知
        dp[index][remain]依赖于index+1行的格子
         */
        int[][] dp = new int[arr.length + 1][aim + 1];
        // base case:if (index == arr.length) return remain == 0 ? 1 : 0;
        dp[arr.length][0] = 1;
        for (int index = arr.length - 1; index >= 0; index--)
            for (int remain = 0; remain <= aim; remain++)
                // index位置的货币,用0张、用1张、用2张......
                for (int i = 0; i * arr[index] <= remain; i++)
                    dp[index][remain] += dp[index + 1][remain- i * arr[index]];
        return dp[0][aim];
    }

    /**
     * 动态规划,枚举优化
     * @param arr
     * @param aim
     * @return
     */
    public static int method04(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        int[][] dp = new int[arr.length + 1][aim + 1];
        dp[arr.length][0] = 1;
        for (int index = arr.length - 1; index >= 0; index--)
            for (int remain = 0; remain <= aim; remain++)
                /*
                观察:
                dp[index][remain]                  = dp[index + 1][remain] + dp[index + 1][remain - 1 * arr[index]] - dp[index + 1][remain - 2 * arr[index]] - ......
                dp[index][remain - 1 * arr[index]] =                         dp[index + 1][remain - 1 * arr[index]] - dp[index + 1][remain - 2 * arr[index]] - ......
                所以:
                dp[index][remain]                  = dp[index + 1][remain] + dp[index][remain - 1 * arr[index]]
                因此省去了for循环的枚举行为
                 */
                dp[index][remain] = remain >= arr[index] ?
                        dp[index + 1][remain] + dp[index][remain - arr[index]] :
                        dp[index + 1][remain];
        return dp[0][aim];
    }

    public static void main(String[] args) {
   
     
        int[] arr = {
   
     5, 10, 50, 100};
        int aim = 1000;
        System.out.println(method01(arr, aim));
        System.out.println(method02(arr, aim));
        System.out.println(method03(arr, aim));
        System.out.println(method04(arr, aim));
    }
}

6、给定一个字符串str,给定一个字符串类型的数组arr,出现的字符都是小写英文。arr每一个字符串,代表一张贴纸,你可以把单个字符剪开使用,目的是拼出str来。返回需要至少多少张贴纸可以完成这个任务。例子:str= “babac”,arr = {“ba”,“c”,“abcd”}。ba + ba + c 3张, abcd + abcd 2张, abcd + ba 2张。所以返回2。

这一题也是先写出暴力递归,就很好改成动态规划,但是只需改出记忆化搜索就可以了,因为它的暴力递归的可变参数,是一个字符串,无法确定范围

/**
 * 给定一个字符串str,给定一个字符串类型的数组arr,出现的字符都是小写英文。
 * arr每一个字符串,代表一张贴纸,你可以把单个字符剪开使用,目的是拼出str来。返回需要至少多少张贴纸可以完成这个任务。
 * 例子:str= "babac",arr = {"ba","c","abcd"}。ba + ba + c 3张, abcd + abcd 2张, abcd + ba 2张。所以返回2。
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming06 {
   
     

    /**
     * 暴力递归
     * @param str 目标字符串
     * @param arr 贴纸数组
     * @return
     */
    public static int getMin01(String str, String[] arr) {
   
     
        int res = process01(str, arr);
        return res == Integer.MAX_VALUE ? -1 : res;
    }

    /**
     * 暴力递归
     * @param str 剩余字符串
     * @param arr 贴纸数组
     * @return
     */
    private static int process01(String str, String[] arr) {
   
     
        // 字符串已经拼好,还需0张贴纸
        if (str.length() == 0) return 0;
        int min = Integer.MAX_VALUE;
        // 尝试每一张贴纸
        for (int i = 0; i < arr.length; i++) {
   
     
            // 减去选择的贴纸的字符,返回新的剩余要拼的字符串
            String newStr = getNewStr(str, arr[i]);
            // 选择的贴纸没有削去任何一个字符,无效,跳过
            if (newStr.length() == str.length()) continue;
            // 剩余字符串newStr,往下继续跑递归
            min = Math.min(min, process01(newStr, arr));
        }
        // 算上自己使用的一张贴纸
        if (min != Integer.MAX_VALUE) min += 1;
        return min;
    }

    /**
     * 原字符串,减去贴纸中的字符,返回新的字符串
     * @param s1 原字符串
     * @param s2 贴纸
     * @return
     */
    private static String getNewStr(String s1, String s2) {
   
     
        char[] chs1 = s1.toCharArray();
        char[] chs2 = s2.toCharArray();
        int[] count = new int[26];
        // 统计原字符串的各种字符数
        for (int i = 0; i < chs1.length; i++) {
   
     
            count[chs1[i] - 'a']++;
        }
        // 减去贴纸含有的各种字符的字符数
        for (int i = 0; i < chs2.length; i++) {
   
     
            count[chs2[i] - 'a']--;
        }
        // 拼接处新的剩余字符串
        StringBuilder sb = new StringBuilder();
        for (int i = 0; i < count.length; i++) {
   
     
            if (count[i] > 0) {
   
     
                for (int j = 0; j < count[i]; j++) {
   
     
                    sb.append((char)(i + 'a'));
                }
            }
        }
        return sb.toString();
    }

    public static int getMin(String str, String[] arr) {
   
     
        /*
        map记录每种贴纸的字符出现的个数
        例如map[i][0] 表示第i张贴纸,a字符出现的次数(a - a == 0)
        初始化号map表
         */
        int[][] map = new int[arr.length][26];
        for (int i = 0; i < arr.length; i++) {
   
     
            char[] chars = arr[i].toCharArray();
            for (char aChar : chars) {
   
     
                map[i][aChar - 'a'] += 1;
            }
        }

        // 缓存表,key剩余字符串,value需要的贴纸数,一个一维表,因为暴力递归中只有一个可变参数(剩余字符串)
        Map<String, Integer> dp = new HashMap<>();
        dp.put("", 0);
        return process(str, map, dp);
    }

    private static int process(String remain, int[][] map, Map<String, Integer> dp) {
   
     
        // 如果缓存表有值,直接返回
        if (dp.containsKey(remain)) return dp.get(remain);

        // 统计原字符串的各种字符数
        char[] chars = remain.toCharArray();
        int[] temp = new int[26];
        for (char aChar : chars) {
   
     
            temp[aChar - 'a'] += 1;
        }
        int res = Integer.MAX_VALUE;

        // 尝试每一张贴纸
        for (int i = 0; i < map.length; i++) {
   
     
            StringBuilder sb = new StringBuilder();
            // 优化,只选择能消除掉当前字符串第一个字符的贴纸(因为第一个字符串总要消掉的,先消后消不会影响结果)
            if (map[i][chars[0] - 'a'] == 0) continue;
            // 减去贴纸含有的各种字符的字符数
            for (int j = 0; j < 26; j++) {
   
     
                for (int k = 0; k < Math.max(0, temp[j] - map[i][j]); k++) {
   
     
                    sb.append((char) (j + 'a'));
                }
            }
            // 剩余字符串newStr,往下继续跑递归
            int next = process(sb.toString(), map, dp);
            // next + 1 算上自己使用的一张贴纸
            if (next != -1) res = Math.min(res, next + 1);
        }

        // 结果返回前,先放入缓存表
        dp.put(remain, res == Integer.MAX_VALUE ? -1 : res);
        return dp.get(remain);
    }

    public static void main(String[] args) {
   
     
        String str = "babac";
        String[] arr = {
   
     "ba","c","abcd"};
        System.out.println(getMin01(str, arr));
        System.out.println(getMin(str, arr));
    }

}

7、求两个字符串的最长公共子序列长度

/**
 * 求两个字符串的最长公共子序列长度
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming07 {
   
     

    /**
     * 暴力递归
     * @param str1
     * @param str2
     * @return
     */
    public static int getMaxCommon01(String str1, String str2) {
   
     
        char[] chars1 = str1.toCharArray();
        char[] chars2 = str2.toCharArray();
        return process01(chars1, chars2, chars1.length - 1, chars2.length - 1);
    }

    /**
     * 求chars1从0到r1与chars2从0到r2的最长公共子序列长度
     * @param chars1
     * @param chars2
     * @param r1
     * @param r2
     * @return
     */
    private static int process01(char[] chars1, char[] chars2, int r1, int r2) {
   
     
        if (r1 == 0 && r2 == 0) {
   
     
            // 只剩一个字符,相等时1,不等是0
            return chars1[r1] == chars2[r2] ? 1 : 0;
        } else if (r1 == 0) {
   
     
            // chars1只剩一个字符,如果和chars2[r2]相等,返回1,不等看chars2前面的字符
            return chars1[r1] == chars2[r2] ? 1 : process01(chars1, chars2, r1, r2 - 1);
        } else if (r2 == 0) {
   
     
            // chars2只剩一个字符,如果和chars1[r1]相等,返回1,不等看chars1前面的字符
            return chars1[r1] == chars2[r2] ? 1 : process01(chars1, chars2, r1 - 1, r2);
        } else {
   
     
            // 不要r1,看公共子序列多长
            int p1 = process01(chars1, chars2, r1 - 1, r2);
            // 不要r2,看公共子序列多长
            int p2 = process01(chars1, chars2, r1, r2 - 1);
            // r1 == r2,看公共子序列多长
            int p3 = chars1[r1] == chars2[r2] ? process01(chars1, chars2, r1 - 1, r2 - 1) : 0;
            // 三个结果PK
            return Math.max(Math.max(p1, p2), p3);
        }
    }

    /**
     * 改成动态规划
     * @param str1
     * @param str2
     * @return
     */
    public static int getMaxCommon(String str1, String str2) {
   
     
        char[] chars1 = str1.toCharArray();
        char[] chars2 = str2.toCharArray();
        /*
        dp[r1][r2]:
        chars1从0到r1与chars2从0到r2的最长公共子序列长度
         */
        int[][] dp = new int[chars1.length][chars2.length];
        // if (r1 == 0 && r2 == 0)
        dp[0][0] = chars1[0] == chars2[0] ? 1 : 0;
        // else if (r1 == 0)
        for (int i = 1; i < chars2.length; i++) {
   
     
            dp[0][i] = Math.max(dp[0][i-1], chars2[i] == chars1[0] ? 1 : 0);
        }
        // else if (r2 == 0)
        for (int i = 1; i < chars1.length; i++) {
   
     
            dp[i][0] = Math.max(dp[i-1][0], chars1[i] == chars2[0] ? 1 : 0);
        }
        // else
        for (int i = 1; i < chars1.length; i++) {
   
     
            for (int j = 1; j < chars2.length; j++) {
   
     
                dp[i][j] = Math.max(dp[i-1][j], dp[i][j-1]);
                if (chars1[i] == chars2[j]) dp[i][j] = Math.max(dp[i][j], dp[i-1][j-1] + 1);
            }
        }
        return dp[chars1.length - 1][chars2.length - 1];
    }

}

8、给定一个数组,代表每个人喝完咖啡准备刷杯子的时间,只有一台咖啡机,一次只能洗一个杯子,时间耗费a,洗完才能洗下一杯,每个咖啡杯也可以自己挥发干净,时间耗费b,咖啡杯可以并行挥发,返回让所有咖啡杯变干净的最早完成时间。三个参数: int[] arr、int a、 int b

/**
 * 给定一个数组,代表每个人喝完咖啡准备刷杯子的时间
 * 只有一台洗咖啡机,一次只能洗一个杯子,时间耗费a,洗完才能洗下一杯
 * 每个咖啡杯也可以自己挥发干净,时间耗费b,咖啡杯可以并行挥发
 * 返回让所有咖啡杯变干净的最早完成时间
 * 三个参数: int[] arr、int a、 int b
 * Created by huangjunyi on 2022/9/4.
 */
public class DynamicProgramming08 {
   
     

    public static int processByRecursive(int[] arr, int a, int b) {
   
     
        return processByRecursive(arr, a, b, 0, 0);
    }

    /**
     * 暴力递归
     * @param arr 每个人喝完咖啡准备刷杯子的时间
     * @param a 洗杯子花费的时间
     * @param b 杯子挥发干净的时间
     * @param index 当前的杯子
     * @param washLine 咖啡机可以使用的时间
     * @return
     */
    private static int processByRecursive(int[] arr, int a, int b, int index, int washLine) {
   
     

        //已经到了最后一个杯子,手洗和挥发,选结束时间早的
        //但是如果选手洗,要看喝完咖啡和洗咖啡机能用的时间,哪个比较晚
        if (index == arr.length - 1) return Math.min(Math.max(arr[index], washLine) + a, arr[index] + b);

        //当前杯子选择使用洗咖啡机,洗完的时间
        int time1 = Math.max(arr[index], washLine) + a;
        //其他杯子也都干净的时间,因为选择了使用洗咖啡机,所以洗咖啡机能用的时间要更新
        int time2 = processByRecursive(arr, a, b, index + 1, time1);
        //两个时间PK
        int p1 = Math.max(time1, time2);

        //当前杯子选择挥发,挥发干净的时间
        int time3 = arr[index] + b;
        //其他杯子也都干净的时间
        int time4 = processByRecursive(arr, a, b, index + 1, washLine);
        //两个时间PK
        int p2 = Math.max(time3, time4);

        //两个时间取最小
        return Math.min(p1, p2);
    }

    /**
     * 动态规划
     * @param arr 每个人喝完咖啡准备刷杯子的时间
     * @param a 洗杯子花费的时间
     * @param b 杯子挥发干净的时间
     * @return
     */
    private static int processByDp(int[] arr, int a, int b) {
   
     
        /*
        应为暴力递归中是两个可变参数,所以改成动态规划是一张二维表
        但是第二个参数washLine有点难估计,属于业务限制模型
        所以就看所有杯子都选择用洗咖啡机洗,最大能冲到什么时间
        */
        int washLineLimit = 0;
        for (int i = 0; i < arr.length; i++) {
   
     
            washLineLimit = Math.max(washLineLimit, arr[i]) + a;
        }
        // dp[i][j] 从i号杯子开始,洗咖啡机j时间点可以用,所有杯子干净需要的时间
        int[][] dp = new int[arr.length][washLineLimit + 1];

        // 根据base case初始化dp表
        // 已经到了最后一个杯子,手洗和挥发,选结束时间早的
        // 但是如果选手洗,要看喝完咖啡和洗咖啡机能用的时间,哪个比较晚
        // if (index == arr.length - 1) return Math.min(Math.max(arr[index], washLine) + a, arr[index] + b);
        for (int i = 0; i <= washLineLimit; i++) {
   
     
            dp[arr.length - 1][i] = Math.min(Math.max(arr[arr.length - 1], i) + a, arr[arr.length - 1] + b);
        }

        for (int i = arr.length - 2; i >= 0; i--) {
   
     
            for (int j = 0; j <= washLineLimit; j++) {
   
     

                // 当前杯子选择用洗咖啡机洗,洗完的时间
                int p1 = Integer.MAX_VALUE;
                int time1 = Math.max(arr[i], j) + a;
                // 判断越界,如果当前杯子选择用洗咖啡机洗,洗完时间超过washLineLimit,是无效情况,跳过
                if (time1 > washLineLimit) continue;
                //其他杯子也都干净的时间,因为选择了使用洗咖啡机,所以洗咖啡机能用的时间要更新
                int time2 = dp[i + 1][time1];
                //两个时间PK
                p1 = Math.max(time1, time2);

                //当前杯子选择挥发,挥发干净的时间
                int time3 = arr[i] + b;
                //其他杯子也都干净的时间
                int time4 = dp[i + 1][j];
                //两个时间PK
                int p2 = Math.max(time3, time4);

                //两个时间取最小
                dp[i][j] = Math.min(p1, p2);
            }
        }
        // return processByRecursive(arr, a, b, 0, 0);
        return dp[0][0];
    }

    public static void main(String[] args) {
   
     
        int[] arr = {
   
     1,1,5,5,7,10,12,12,12,12,12,12,15};
        int a=3;
        int b=10;
        System.out.println(processByRecursive(arr, a, b));
        System.out.println(processByDp(arr, a, b));
    }

}

9、马踏棋盘问题

  • 规定马从0,0位置出发,棋盘是10*9大小的棋盘
  • 给定三个整形变量x,y,k,表示马走k步到达x,y位置
  • 问有几种走法
/**
 * 马踏棋盘问题
 * 规定马从0,0位置出发,棋盘是10*9大小的棋盘
 * 给定三个整形变量x,y,k,表示马走k步到达x,y位置
 * 问有几种走法
 * Created by huangjunyi on 2022/9/4.
 */
public class DynamicProgramming09 {
   
     

    public static int processByRecursive(int x, int y, int k) {
   
     
        // 越界返回0
        if (x < 0 || x >= 10 || y < 0 || y >= 9) return 0;
        // 剩余0步要走,如果到达目标位置,返回1,代表一种走法,否则返回0
        if (k == 0) return x == 0 && y == 0 ? 1 : 0;
        // 从(x,y)位置,反方向走,往8个方向跳,累加跳回到起点的方法数,就等于从起点跳到(x,y)的方法数
        return  processByRecursive(x + 2, y + 1, k - 1) +
                processByRecursive(x + 1, y + 2, k - 1) +
                processByRecursive(x - 1, y + 2, k - 1) +
                processByRecursive(x - 2, y + 1, k - 1) +
                processByRecursive(x - 2, y - 1, k - 1) +
                processByRecursive(x - 1, y - 2, k - 1) +
                processByRecursive(x + 1, y - 2, k - 1) +
                processByRecursive(x + 2, y - 1, k - 1);
    }

    public static int processByDp(int x, int y, int k) {
   
     
        // 三个变化参数,所以准备一张3维表
        int[][][] dp = new int[10][9][k + 1];
        // base case:if (k == 0) return x == 0 && y == 0 ? 1 : 0;
        dp[0][0][0] = 1;
        /*
        观察暴力递归
        剩k步时的答案,依赖剩k-1步时的答案
        所以上层答案依赖下一层,不依赖本层
        已就是dp[i][j][k]依赖dp[x][y][k-1]
         */
        for (int level = 1; level <= k; level++) {
   
     
            for (int i = 0; i < 10; i++) {
   
     
                for (int j = 0; j < 9; j++) {
   
     
                    dp[i][j][level] =
                            // 从(x,y)位置,反方向走,往8个方向跳,累加跳回到起点的方法数,就等于从起点跳到(x,y)的方法数
                            getFromDp(dp, i + 2, j + 1, level - 1) +
                            getFromDp(dp, i + 1, j + 2, level - 1) +
                            getFromDp(dp, i - 1, j + 2, level - 1) +
                            getFromDp(dp, i - 2, j + 1, level - 1) +
                            getFromDp(dp, i - 2, j - 1, level - 1) +
                            getFromDp(dp, i - 1, j - 2, level - 1) +
                            getFromDp(dp, i + 1, j - 2, level - 1) +
                            getFromDp(dp, i + 2, j - 1, level - 1);
                }
            }
        }
        return dp[x][y][k];
    }

    public static int getFromDp(int[][][] dp, int x, int y, int k) {
   
     
        if (x < 0 || x >= 10 || y < 0 || y >= 9) return 0;
        return dp[x][y][k];
    }

    public static void main(String[] args) {
   
     
        System.out.println(processByRecursive(6, 8, 10));
        System.out.println(processByDp(6, 8, 10));
    }

}

10、最长回文子序列

力扣516题
https://leetcode.cn/problems/longest-palindromic-subsequence/description/
给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。
子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。

/**
 * https://leetcode.cn/problems/longest-palindromic-subsequence/
 * 给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。
 * 子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。
 * Created by huangjunyi on 2022/11/27.
 */
public class DynamicProgramming10 {
   
     

    /**
     * 动态规划
     */
    class Solution {
   
     
        public int longestPalindromeSubseq(String s) {
   
     
            char[] chs = s.toCharArray();
            int N = chs.length;
            // l和r两个可变参数,l和r的变化返回都是0~N-1
            int[][] dp = new int[N][N];
            /*
            初始化dp表的两种情况
            剩一个字符
            剩两个字符
            if (l == r) return 1;
            if (l == r - 1) return chs[l] == chs[r] ? 2 : 0;
            */
            dp[N-1][N-1] = 1;
            for (int i = 0; i < N - 1; i++) {
   
     
                dp[i][i] = 1;
                dp[i][i + 1] = chs[i] == chs[i + 1] ? 2 : 1;
            }
            for (int l = N - 2; l >= 0; l--) {
   
     
                for (int r = l + 2; r < N; r++) {
   
     
                    // 不要l字符和r字符
                    int p1 = dp[l + 1][r - 1];
                    // 不要l字符
                    int p2 = dp[l + 1][r];
                    // 不要r字符
                    int p3 = dp[l][r - 1];
                    // l字符和r字符相等,要
                    int p4 = chs[l] == chs[r] ? 2 + dp[l + 1][r - 1] : 0;
                    // 4种情况PK一下
                    dp[l][r] =  Math.max(Math.max(p1, p2), Math.max(p3, p4));
                }
            }
            // return process(chs, 0, chs.length - 1);
            return dp[0][N - 1];
        }
    }
    /**
     * 暴力递归版本
     */
    /*class Solution {
        public int longestPalindromeSubseq(String s) {
            char[] chs = s.toCharArray();
            return process(chs, 0, chs.length - 1);
        }

        private int process(char[] chs, int l, int r) {
            // 剩一个字符
            if (l == r) return 1;
            // 剩两个字符
            if (l == r - 1) return chs[l] == chs[r] ? 2 : 0;
            // 不要l字符和r字符
            int p1 = process(chs, l + 1, r - 1);
            // 不要l字符
            int p2 = process(chs, l + 1, r);
            // 不要r字符
            int p3 = process(chs, l, r - 1);
            // l字符和r字符相等,要
            int p4 = chs[l] == chs[r] ? 2 + process(chs, l + 1, r - 1) : 0;
            // 4种情况PK一下
            return Math.max(Math.max(p1, p2), Math.max(p3, p4));
        }
    }*/
}

11、最小距离累加和

给定一个二维数组matrix,一个人必须从左上角出发,最后达到右下角
沿途只可以向下或者向右走,沿途的数字都累加就是距离累加和
返回最小距离累加和

/**
 * 给定一个二维数组matrix,一个人必须从左上角出发,最后达到右下角
 * 沿途只可以向下或者向右走,沿途的数字都累加就是距离累加和
 * 返回最小距离累加和
 * Created by huangjunyi on 2022/11/27.
 */
public class DynamicProgramming11 {
   
     

    /**
     * 动态规划
     * @param matrix
     * @return
     */
    public static int getMinPathSum01(int[][] matrix) {
   
     
        if (matrix == null || matrix.length == 0 || matrix[0] == null || matrix[0].length == 0) return 0;
        int N = matrix.length;
        int M = matrix[0].length;
        // dp[i][j] 走左上角出发,走到i行j列的格子,最小距离累加和是多少
        int[][] dp = new int[N][M];
        // 初始化第0行
        dp[0][0] = matrix[0][0];
        for (int i = 1; i < M; i++) {
   
     
            dp[0][i] = dp[0][i - 1] + matrix[0][i];
        }
        // dp[i][j] 从左边走过来(dp[i][j-1]),从上边走过来(dp[i-1][j]),两个方向,选最路径和最短
        for (int i = 1; i < N; i++) {
   
     
            dp[i][0] = dp[i - 1][0] + matrix[i][0];
            for (int j = 1; j < M; j++) {
   
     
                dp[i][j] = Math.min(dp[i][j-1], dp[i-1][j]) + matrix[i][j];
            }
        }
        return dp[N - 1][M - 1];
    }

    /**
     * 动态规划,空间压缩
     * @param matrix
     * @return
     */
    public static int getMinPathSum02(int[][] matrix) {
   
     
        if (matrix == null || matrix.length == 0 || matrix[0] == null || matrix[0].length == 0) return 0;
        int N = matrix.length;
        int M = matrix[0].length;
        // 因为dp[i][j]只依赖左边和上边一个格子,所以不需要二维dp表,一位表滚动更新即可
        int[] dp = new int[M];
        // 初始化第0行
        dp[0] = matrix[0][0];
        for (int i = 1; i < M; i++) {
   
     
            dp[i] = dp[i - 1] + matrix[0][i];
        }
        /*
        假设滚动更新到i行,现在要填dp[j],此时还没更新dp[j],那么:
        dp[j-1] 对应 dp[i][j-1]
        dp[j] 对应 dp[i-1][j]

        dp[j] = Math.min(dp[j-1], dp[j]) + matrix[i][j];
         */
        for (int i = 1; i < N; i++) {
   
     
            dp[0] = dp[0] + matrix[i][0];
            for (int j = 1; j < M; j++) {
   
     
                dp[j] = Math.min(dp[j-1], dp[j]) + matrix[i][j];
            }
        }
        return dp[M - 1];
    }

}

12、货币组合问题2

货币组合问题2
给定一个整形数组arr,表示一组货币
数组每个值都是一张货币
但是值相同的货币没有任何不同
给定一个整形aim,表示需要通过不同的货币拼凑一起到总额aim
从arr中取货币
求有多少种组合方法

例如:arr = [1,2,1,1,2,1,2],aim = 4
方法:1+1+1+1、1+1+2、2+2

/**
 * 货币组合问题2
 * 给定一个整形数组arr,表示一组货币
 * 数组每个值都是一张货币
 * 但是值相同的货币没有任何不同
 * 给定一个整形aim,表示需要通过不同的货币拼凑一起到总额aim
 * 从arr中取货币
 * 求有多少种组合方法
 *
 * 例如:arr = [1,2,1,1,2,1,2],aim = 4
 * 方法:1+1+1+1 1+1+2 2+2
 * Created by huangjunyi on 2022/9/3.
 */
public class DynamicProgramming12 {
   
     

    /**
     * 暴力递归
     * @param arr 货币数组
     * @param aim 目标总额
     * @return 方法数
     */
    public static int method01(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        // key:货币面值,value:货币张数
        Map<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < arr.length; i++) {
   
     
            if (map.containsKey(arr[i])) map.put(arr[i], map.get(arr[i]) + 1);
            else map.put(arr[i], 1);
        }
        // 货币面值数组
        int[] money = new int[map.size()];
        // 货币张数数组
        int[] count = new int[map.size()];
        int index = 0;
        // 根据map初始化money和count数组
        for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
   
     
            money[index] = entry.getKey();
            count[index] = entry.getValue();
            index++;
        }
        return process01(money, count, 0, aim);
    }

    /**
     * 暴力递归
     * @param money 货币面值数组
     * @param count 货币张数数组
     * @param index 第index号货币开始,往后的货币,自由选择
     * @param rest 剩余要凑够的钱
     * @return 方法数
     */
    private static int process01(int[] money, int[] count, int index, int rest) {
   
     
        if (index == money.length) return rest == 0 ? 1 : 0;
        int res = 0;
        // index号货币,使用0张,使用1张,使用2张,......,使用count[index]张
        for (int i = 0; rest - money[index] * i >= 0 && i <= count[index]; i++) {
   
     
            res += process01(money, count, index + 1, rest - money[index] * i);
        }
        return res;
    }

    /**
     * 动态规划
     * @param arr 货币数组
     * @param aim 目标总额
     * @return 方法数
     */
    public static int method02(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        // key:货币面值,value:货币张数
        Map<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < arr.length; i++) {
   
     
            if (map.containsKey(arr[i])) map.put(arr[i], map.get(arr[i]) + 1);
            else map.put(arr[i], 1);
        }
        // 货币面值数组
        int[] money = new int[map.size()];
        // 货币张数数组
        int[] count = new int[map.size()];
        int index = 0;
        // 根据map初始化money和count数组
        for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
   
     
            money[index] = entry.getKey();
            count[index] = entry.getValue();
            index++;
        }

        /*
        暴力递归2个可变参数,二维表,观察可变参数范围,确定dp表的长度
        dp[index][rest] index号货币开始往后自由选择,凑够rest元,的方法数
         */
        int[][] dp = new int[money.length + 1][aim + 1];
        // 根据 base case 初始化dp表:if (index == money.length) return rest == 0 ? 1 : 0;
        dp[money.length][0] = 1;

        /*
        index号货币,使用0张,使用1张,使用2张,......,使用count[index]张
        for (int i = 0; rest - money[index] * i >= 0 && i <= count[index]; i++) {
            res += process01(money, count, index + 1, rest - money[index] * i);
        }
        */
        for (index = money.length - 1; index >= 0; index--) {
   
     
            for (int rest = 0; rest <= aim; rest++) {
   
     
                int res = 0;
                for (int i = 0; rest - money[index] * i >= 0 && i <= count[index]; i++) {
   
     
                    res += dp[index + 1][rest - money[index] * i];
                }
                dp[index][rest] = res;
            }
        }
        // return process01(money, count, 0, aim);
        return dp[0][aim];
    }

    /**
     * 动态规划 枚举优化
     * @param arr 货币数组
     * @param aim 目标总额
     * @return 方法数
     */
    public static int method03(int[] arr, int aim) {
   
     
        if (arr == null || arr.length == 0 || aim < 0) return 0;
        Map<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < arr.length; i++) {
   
     
            if (map.containsKey(arr[i])) map.put(arr[i], map.get(arr[i]) + 1);
            else map.put(arr[i], 1);
        }
        int[] money = new int[map.size()];
        int[] count = new int[map.size()];
        int index = 0;
        for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
   
     
            money[index] = entry.getKey();
            count[index] = entry.getValue();
            index++;
        }

        int[][] dp = new int[money.length + 1][aim + 1];
        dp[money.length][0] = 1;
        for (index = money.length - 1; index >= 0; index--) {
   
     
            for (int rest = 0; rest <= aim; rest++) {
   
     
                /*
                观察:
                假设count[index] = 3,index号货币有3张
                dp[index][rest]                    = dp[index + 1][rest] + dp[index + 1][rest - 1 * money[index]] + dp[index + 1][rest - 2 * money[index]] + dp[index + 1][rest - 3 * money[index]]
                dp[index][rest - 1 * money[index]] =                       dp[index + 1][rest - 1 * money[index]] + dp[index + 1][rest - 2 * money[index]] + dp[index + 1][rest - 3 * money[index]] + dp[index + 1][rest - 4 * money[index]]
                所以:
                dp[index][rest]                    = dp[index + 1][rest] + dp[index][rest - 1 * money[index]] - dp[index + 1][rest - 4 * money[index]];
                dp[index][rest]                    = dp[index + 1][rest] + dp[index][rest - 1 * money[index]] - dp[index + 1][rest - (count[index] + 1) * money[index]];
                 */
                dp[index][rest] = dp[index + 1][rest];
                if (rest - money[index] >= 0) dp[index][rest] += dp[index][rest - money[index]];
                if (rest - (count[index] + 1) * money[index] >= 0) dp[index][rest] -= dp[index + 1][rest - (count[index] + 1) * money[index]];
            }
        }

        return dp[0][aim];
    }

    public static void main(String[] args) {
   
     
        int[] arr = {
   
     1,2,1,1,2,1,2};
        int aim = 4;
        System.out.println(method01(arr, aim));
        System.out.println(method02(arr, aim));
        System.out.println(method03(arr, aim));
    }
}

13、英雄打怪兽

给定三个参数,N、M、K
怪兽有N滴血,等着英雄来砍自己
英雄每一次打击,都会让怪兽流失[0-M]的血量
到底流失多少?每一次在[0-M]上等概率的获得一个值
求K次打击之后,英雄把怪兽砍死的概率

/**
 * 给定三个参数,N、M、K
 * 怪兽有N滴血,等着英雄来砍自己
 * 英雄每一次打击,都会让怪兽流失[0-M]的血量
 * 到底流失多少?每一次在[0-M]上等概率的获得一个值
 * 求K次打击之后,英雄把怪兽砍死的概率
 * Created by huangjunyi on 2022/11/28.
 */
public class DynamicProgramming13 {
   
     

    /**
     * 暴力递归
     * @param N 怪兽有N滴血
     * @param M 怪兽流失血量范围
     * @param K 英雄打击K次
     * @return
     */
    public static double killMonster01(int N, int M, int K) {
   
     
        if (N < 1 || M < 1 || K < 1) return 0;
        // 所有情况数
        long all = (long) Math.pow(M + 1, K);
        // 砍死怪兽的情况数
        long kill = process01(N, M, K);
        return ((double) kill) / ((double) all);
    }

    /**
     * 暴力递归
     * @param hp 怪兽剩余血量
     * @param M 怪兽流失血量范围
     * @param rest 英雄剩K次要砍
     * @return
     */
    private static long process01(int hp, int M, int rest) {
   
     
        // base case:砍完了,怪兽血量减到0以下,返回1中有效方法数
        if (rest == 0) return hp <= 0 ? 1 : 0;
        // 怪兽已经死了,也要继续砍,因为往后的都是算入有效方法数的
        if (hp <= 0) return (long) Math.pow(M + 1, rest);
        // 枚举砍出的伤害值,累加方法数
        long ways = 0;
        for (int i = 0; i <= M; i++) {
   
     
            ways += process01(hp - i, M, rest - 1);
        }
        return ways;
    }

    /**
     * 动态规划
     * @param N 怪兽有N滴血
     * @param M 怪兽流失血量范围
     * @param K 英雄打击K次
     * @return
     */
    public static double killMonster02(int N, int M, int K) {
   
     
        if (N < 1 || M < 1 || K < 1) return 0;
        // 所有情况数
        long all = (long) Math.pow(M + 1, K);
        /*
        暴力递归两个可变参数,剩余要砍次数、怪兽剩余血量,所以是一张二维dp表
        dp[rest][hp]
        英雄剩余要看rest次,怪兽剩余血量hp,砍死怪兽的方法数
        观察暴力递归的参数范围,可得dp表长度
         */
        long[][] dp = new long[K + 1][N + 1];
        /*
        根据base case,初始化dp表:
        if (rest == 0) return hp <= 0 ? 1 : 0;
        if (hp <= 0) return (long) Math.pow(M + 1, rest);
         */
        dp[0][0] = 1;
        for (int rest = 1; rest <= K; rest++) {
   
     
            dp[rest][0] =  (long) Math.pow(M + 1, rest);
        }
        /*
         观察暴力递归中,外层递归与里层递归的依赖关系
         确定dp表的填表顺序
         long ways = 0;
         for (int i = 0; i < M; i++) {
             ways += process01(hp - i, M, rest - 1);
         }
         dp[rest][...] 依赖 dp[rest - 1][...]
         所以是从上往下填
          */
        for (int rest = 1; rest <= K; rest++) {
   
     
            for (int hp = 1; hp <= N; hp++) {
   
     
                long ways = 0;
                for (int i = 0; i <= M; i++) {
   
     
                    // 这里要注意hp减完有可能越界,越界了就通过公式算
                    ways += hp - i >= 0 ? dp[rest - 1][hp - i] : (long) Math.pow(M + 1, rest - 1);
                }
                dp[rest][hp] = ways;
            }
        }
        /*
        根据暴力递归的最外层递归,确定返回值
        long kill = process01(N, M, K);
        return ((double) kill) / ((double) all);
         */
        return ((double) dp[K][N]) / ((double) all);
    }

    /**
     * 动态规划 枚举行为优化
     * @param N 怪兽有N滴血
     * @param M 怪兽流失血量范围
     * @param K 英雄打击K次
     * @return
     */
    public static double killMonster03(int N, int M, int K) {
   
     
        if (N < 1 || M < 1 || K < 1) return 0;
        long all = (long) Math.pow(M + 1, K);
        long[][] dp = new long[K + 1][N + 1];
        dp[0][0] = 1;
        for (int rest = 1; rest <= K; rest++) {
   
     
            dp[rest][0] =  (long) Math.pow(M + 1, rest);
        }
        for (int rest = 1; rest <= K; rest++) {
   
     
            for (int hp = 1; hp <= N; hp++) {
   
     
//                long ways = 0;
//                for (int i = 0; i < M; i++) {
   
     
//                    // 这里要注意hp减完有可能越界,越界了就通过公式算
//                    ways += hp - i >= 0 ? dp[hp - i][rest - 1] : (long) Math.pow(M + 1, rest - 1);
//                }
//                dp[rest][hp] = ways;

                /*
                有枚举行为,就随便算一个格子看依赖关系就行了
                观察dp[5][10],M=5,英雄剩5次要砍,怪兽剩10点血
                dp[5][10] = dp[4][5...10]
                dp[5][11] = dp[4][6...11]
                所以:
                dp[5][11] = dp[4][11] + dp[5][10] - dp[4][5]
                推断出:
                dp[rest][hp] = dp[rest - 1][hp] + dp[rest][hp - 1] - dp[rest - 1][hp - M - 1]
                但是那是hp - M - 1不越界的情况,如果越界:
                dp[rest][hp] = dp[rest - 1][hp] + dp[rest][hp - 1] - (long) Math.pow(M + 1, rest - 1)
                 */
                dp[rest][hp] = dp[rest][hp - 1] + dp[rest - 1][hp];
                if (hp - 1 - M >= 0) {
   
     
                    dp[rest][hp] -= dp[rest - 1][hp - 1 - M];
                } else {
   
     
                    dp[rest][hp] -= Math.pow(M + 1, rest - 1);
                }
            }
        }
        return ((double) dp[K][N]) / ((double) all);
    }

    public static void main(String[] args) {
   
     
        System.out.println("begin");
        for (int i = 0; i < 20000; i++) {
   
     
            int N = (int) (Math.random() * 10);
            int M = (int) (Math.random() * 10);
            int K = (int) (Math.random() * 10);
            double res1 = killMonster01(N, M, K);
            double res2 = killMonster02(N, M, K);
            double res3 = killMonster03(N, M, K);
            if (res1 != res2 || res1 != res3) {
   
     
                System.out.println("error: " + " N=" + N + " M=" + M + " K=" + K);
                System.out.println("res1=" + res1 + " res2=" + res2 + " res3=" + res3);
                break;
            }
        }
        System.out.println("finished");
    }

}

14、货币组合问题3

arr是面值数组,其中的值都是正数且没有重复。
再给定一个正数aim。
每个值都认为是一种面值,且认为张数是无限的。
返回组成aim的最少货币数。

/**
 * 货币组合问题3
 * arr是面值数组,其中的值都是正数且没有重复。
 * 再给定一个正数aim。
 * 每个值都认为是一种面值,且认为张数是无限的。
 * 返回组成aim的最少货币数。
 *
 * Created by huangjunyi on 2022/11/30.
 */
public class DynamicProgramming14 {
   
     

    /**
     * 暴力递归
     */
    public static int minCoins01(int[] arr, int aim) {
   
     
        return process(arr, 0, aim);
    }

    /**
     * 暴力递归
     * @param arr 货币数组
     * @param index 当前货币
     * @param rest 剩余要凑够的钱数
     * @return
     */
    private static int process(int[] arr, int index, int rest) {
   
     
        // base case 货币都决定完了,剩余要凑够的钱数为0,返回0(代表凑够0元需要0张货币),否则返回MAX,代表无效
        if (index == arr.length) return rest == 0 ? 0 : Integer.MAX_VALUE;
        int min = Integer.MAX_VALUE;
        // index号货币要0张,要1张,要2张......拿最少值
        for (int i = 0; i * arr[index] <= rest; i++) {
   
     
            int next = process(arr, index + 1, rest - i * arr[index]);
            if (next != Integer.MAX_VALUE) min = Math.min(min, next + i);
        }
        return min;
    }

    /**
     * 动态规划
     * @param arr 货币数组
     * @param aim 要凑够的目标货币数
     * @return
     */
    public static int minCoins02(int[] arr, int aim) {
   
     
        /*
        根据暴力递归,可变参数的个数,范围,推测dp表的结构
        可变参数:index rest => 二维dp表
        范围:index:0~arr.length rest: 0~aim
        dp[index][rest] 从index号货币开始往后自由选择,凑出rest元的最少货币数
         */
        int[][] dp = new int[arr.length + 1][aim + 1];
        // 根据暴力递归 base case 初始化dp表
        // if (index == arr.length) return rest == 0 ? 1 : Integer.MAX_VALUE;
        dp[arr.length][0] = 0;
        for (int i = 1; i <= aim; i++) {
   
     
            dp[arr.length][i] = Integer.MAX_VALUE;
        }
        /*
        观察暴力递归外层与内层的依赖关系,确定填表的方向

        int min = Integer.MAX_VALUE;
        // index号货币要0张,要1张,要2张......拿最少值
        for (int i = 0; i * arr[index] <= rest; i++) {
            int next = process(arr, index + 1, rest - i * arr[index]);
            if (next != Integer.MAX_VALUE) min = Math.min(min, next + i);
        }
        index 依赖 index + 1
        所以从下往上填
         */
        for (int index = arr.length - 1; index >= 0; index--) {
   
     
            for (int rest = 1; rest <= aim; rest++) {
   
     
                int min = Integer.MAX_VALUE;
                // index号货币要0张,要1张,要2张......拿最少值
                for (int i = 0; i * arr[index] <= rest; i++) {
   
     
                    int next = dp[index + 1][rest - i * arr[index]];
                    if (next != Integer.MAX_VALUE) min = Math.min(min, next + i);
                }
                dp[index][rest] = min;
            }
        }
        // 根据暴力递归最外层递归的参数,确定返回值
        // return process(arr, 0, aim);
        return dp[0][aim];
    }

    /**
     * 动态规划 枚举行为优化
     * @param arr 货币数组
     * @param aim 要凑够的目标货币数
     * @return
     */
    public static int minCoins03(int[] arr, int aim) {
   
     
        int[][] dp = new int[arr.length + 1][aim + 1];
        dp[arr.length][0] = 0;
        for (int i = 1; i <= aim; i++) {
   
     
            dp[arr.length][i] = Integer.MAX_VALUE;
        }
        for (int index = arr.length - 1; index >= 0; index--) {
   
     
            for (int rest = 1; rest <= aim; rest++) {
   
     
                /*int min = Integer.MAX_VALUE;
                for (int i = 0; i * arr[index] <= rest; i++) {
                    int next = dp[index + 1][rest - i * arr[index]];
                    if (next != Integer.MAX_VALUE) min = Math.min(min, next + i);
                }
                dp[index][rest] = min;*/

                /*
                观察dp[5][10] 假设arr[5]=3:
                dp[5][10] = Math.min(dp[6][10], dp[6][7] + 1, dp[6][4] + 2, dp[6][1] + 3)
                dp[5][7]  = Math.min(           dp[6][7],     dp[6][4] + 1, dp[6][1] + 2)
                所以:
                dp[5][10] = Math.min(dp[6][10], dp[5][7] + 1)
                dp[index][rest] = Math.min(dp[index + 1][rest], dp[index][rest- arr[index]] + 1)
                 */

                dp[index][rest] = dp[index + 1][rest];
                if (rest- arr[index] >= 0 && dp[index][rest- arr[index]] != Integer.MAX_VALUE) {
   
     
                    dp[index][rest] = Math.min(dp[index][rest], dp[index][rest- arr[index]] + 1);
                }
            }
        }
        return dp[0][aim];
    }

}

15、数字切分

给你一个数字num,将它切开
比如给你7,你可以切成2/2/3,也可以切成3/4,但是不能切成3/2/2或4/3,也就是前面的数不能大于后面的数
返回一个数一共有多少中切分方法

/**
 * 数字切分
 * 给你一个数字num,将它切开
 * 比如给你7,你可以切成2/2/3,也可以切成3/4,但是不能切成3/2/2或4/3,也就是前面的数不能大于后面的数
 * 返回一个数一共有多少中切分方法
 * Created by huangjunyi on 2022/11/30.
 */
public class DynamicProgramming15 {
   
     

    /**
     * 暴力递归
     * @param n 要切分的数字
     */
    public static int splitNumber01(int n) {
   
     
        if (n == 0) return 0;
        if (n == 1) return 1;
        return process(1, n);
    }

    /**
     * 暴力递归
     * 前一个数字是preNum,剩下要切分的数字是restNum,切分restNum的切分方法数
     * @param preNum 前一个数字
     * @param restNum 剩下要切分的数字
     * @return
     */
    private static int process(int preNum, int restNum) {
   
     
        // restNum切成了0,有效切分方法数1
        if (restNum == 0)return 1;
        // 前一个数比restNum,无效切分
        if (preNum > restNum) return 0;
        // 当前切出preNum,切出preNum+1,切出preNum+2......,方法数累加
        int ways = 0;
        for (int i = preNum; i <= restNum; i++) {
   
     
            ways += process(i, restNum - i);
        }
        return ways;
    }

    /**
     * 动态规划
     * @param n 要切分的数字
     * @return
     */
    private static int splitNumber02(int n) {
   
     
        /*
        观察暴力递归的可变参数个数,已经可变参数范围
        preNum:1~n
        restNum:0~n
        dp[preNum][restNum]
        前一个数字是preNum,剩下要切分的数字是restNum,切分restNum的切分方法数
         */
        int[][] dp = new int[n + 1][n + 1];
        /*
        根据暴力递归的base case初始化dp表
        if (restNum == 0)return 1;
        if (preNum > restNum) return 0;
         */
        for (int preNum = 1; preNum <= n; preNum++) {
   
     
            dp[preNum][0] = 1;
        }
        /*
        观察暴力递归的外层递归和内层递归的依赖关系,确定dp表的填写顺序
        int ways = 0;
        for (int i = preNum; i <= restNum; i++) {
            ways += process(i, restNum - i);
        }
        return ways;
        preNum依赖preNum+1、preNum+2......
        所以从下往上填

        因为preNum > restNum 都是无效的,所以只需要填上半部分
         */
        for (int preNum = n; preNum >= 1; preNum--) {
   
     
            for (int restNum = preNum; restNum <= n; restNum++) {
   
     
                int ways = 0;
                for (int i = preNum; i <= restNum; i++) {
   
     
                    ways += dp[i][restNum - i];
                }
                dp[preNum][restNum] = ways;
            }
        }
        // 根据暴力递归的最外层递归确定返回值
        // return process(1, n);
        return dp[1][n];
    }

    /**
     * 动态规划 枚举行为优化
     * @param n 要切分的数字
     * @return
     */
    private static int splitNumber03(int n) {
   
     
        int[][] dp = new int[n + 1][n + 1];
        for (int preNum = 1; preNum <= n; preNum++) {
   
     
            dp[preNum][0] = 1;
        }
        for (int preNum = n; preNum >= 1; preNum--) {
   
     
            for (int restNum = preNum; restNum <= n; restNum++) {
   
     
                /*int ways = 0;
                for (int i = preNum; i <= restNum; i++) {
                    ways += dp[i][restNum - i];
                }
                dp[preNum][restNum] = ways;*/

                /*
                观察dp[2][10]
                dp[2][10] = dp[2][8] + dp[3][7] + dp[4][6] + dp[5][5]
                dp[3][10] =            dp[3][7] + dp[4][6] + dp[5][5]
                所以:
                dp[2][10] = dp[2][8] + dp[3][10]
                dp[preNum][restNum] = dp[preNum][restNum - preNum] + dp[preNum + 1][restNum]
                 */
                dp[preNum][restNum] = dp[preNum][restNum - preNum];
                if (preNum != n) dp[preNum][restNum] += dp[preNum + 1][restNum];
            }
        }
        return dp[1][n];
    }

    public static void main(String[] args) {
   
     
        int n = 68;
        System.out.println(splitNumber01(n));
        System.out.println(splitNumber02(n));
        System.out.println(splitNumber03(n));
    }
}

16、较小集合的累加和

给定一个正数数组arr
请把arr中所有的数分成两个集合,尽量让两个集合的累加和接近
返回: 最接近的情况下,较小集合的累加和

/**
 * 较小集合的累加和
 * 给定一个正数数组arr
 * 请把arr中所有的数分成两个集合,尽量让两个集合的累加和接近
 * 返回:
 * 最接近的情况下,较小集合的累加和
 * Created by huangjunyi on 2022/11/30.
 */
public class DynamicProgramming16 {
   
     

    /**
     * 暴力递归
     */
    public static int smallSum01(int[] arr) {
   
     
        if (arr == null || arr.length <= 1) return 0;
        int sum = 0;
        for (int num : arr) {
   
     
            sum += num;
        }
        // arr数组累加和除以2,把题意转换为从arr数组自由选择,
        // 返回最接近但是不超过arr累加和一半的集合累加和
        return process(arr, 0 , sum / 2);
    }

    /**
     * 暴力递归,把题意转换为从index位置,arr后面的数只有做选择,返回最近但是不超过rest的数
     */
    private static int process(int[] arr, int index, int rest) {
   
     
        // base case 没有数需要选择了,返回0
        if (index == arr.length) return 0;
        // 不要index位置的数
        int p1 = process(arr, index + 1, rest);
        // 要index位置的数,但是要判断不能超过rest
        int p2 = 0;
        if (arr[index] <= rest) {
   
     
            p2 = arr[index] + process(arr, index + 1, rest - arr[index]);
        }
        return Math.max(p1, p2);
    }

    public static int smallSum02(int[] arr) {
   
     
        if (arr == null || arr.length <= 1) return 0;
        /*
        可变参数index,rest
        index:0 ~ arr.length
        rest: 0 ~ sum/2
         */
        int sum = 0;
        for (int num : arr) sum += num;
        int[][] dp = new int[arr.length + 1][(sum / 2) + 1];
        // base case: if (index == arr.length) return 0;
        // 数组初始值本身就是0,不用初始化

        /*
        暴力递归中外层与内层的依赖关系
        // 不要index位置的数
        int p1 = process(arr, index + 1, rest);
        // 要index位置的数,但是要判断不能超过rest
        int p2 = 0;
        if (arr[index] <= rest) {
            p2 = arr[index] + process(arr, index + 1, rest - arr[index]);
        }
        return Math.max(p1, p2);
        所以index依赖index+1
        所以从下往上填
         */
        for (int index = arr.length - 1; index >= 0; index--) {
   
     
            for (int rest = 0; rest <= sum / 2; rest++) {
   
     
                // 不要index位置的数
                int p1 = dp[index + 1][rest];
                // 要index位置的数,但是要判断不能超过rest
                int p2 = 0;
                if (arr[index] <= rest) {
   
     
                    p2 = arr[index] + dp[index + 1][rest - arr[index]];
                }
                dp[index][rest] = Math.max(p1, p2);
            }
        }
        // 暴力递归最外层:return process(arr, 0 , sum / 2);
        return dp[0][sum / 2];
    }

}

17、较小集合的累加和2

给定一个正整数数组arr,请把arr中所有的数分成两个集合
如果arr长度为偶数,两个集合包含数的个数要一样多
如果arr长度为奇数,两个集合包含数的个数必须只差一个
请尽量让两个集合的累加和接近
返回: 最接近的情况下,较小结合的累加和

/**
 * 给定一个正整数数组arr,请把arr中所有的数分成两个集合
 * 如果arr长度为偶数,两个集合包含数的个数要一样多
 * 如果arr长度为奇数,两个集合包含数的个数必须只差一个
 * 请尽量让两个集合的累加和接近
 * 返回:
 * 最接近的情况下,较小结合的累加和
 * Created by huangjunyi on 2022/11/30.
 */
public class DynamicProgramming17 {
   
     
    /**
     * 暴力递归
     */
    public static int smallSum01(int[] arr) {
   
     
        if (arr == null || arr.length <= 1) return 0;
        int sum = 0;
        for (int num : arr) {
   
     
            sum += num;
        }
        if ((arr.length & 1) == 0) {
   
     
            return process(arr, 0, arr.length / 2, sum / 2);
        } else {
   
     
            return Math.max(process(arr, 0, arr.length / 2, sum / 2), process(arr, 0, arr.length / 2 + 1, sum / 2));
        }
    }

    /**
     * 从index位置开始,在数组arr中只有挑选数字,必须挑满picks个,
     * 返回最接近rest但是不超过rest的集合累加和
     */
    private static int process(int[] arr, int index, int picks, int rest) {
   
     
        // base case:没数了,看挑完没,没挑完要返回-1,表示无效
        if (index == arr.length) return picks == 0 ? 0 : -1;
        // index的数不要
        int p1 = process(arr, index + 1, picks, rest);
        // index的数要
        int p2 = -1;
        int next = -1;
        if (arr[index] <= rest) next = process(arr, index + 1, picks - 1, rest - arr[index]);
        if (next != -1) p2 = arr[index] + next;
        // 两个累加和PK一下
        return Math.max(p1, p2);
    }

    /**
     * 动态规划
     */
    public static int smallSum02(int[] arr) {
   
     
        if (arr == null || arr.length <= 1) return 0;
        int sum = 0;
        for (int num : arr) {
   
     
            sum += num;
        }

        /*
        可变参数:index picks rest
        index:0 ~ arr.length
        picks: 偶数 0 ~ arr.length / 2 奇数 0 ~ arr.length / 2 + 1
        rest: 0 ~ sum / 2
        dp[index][picks][rest]:从index位置开始,在数组arr中只有挑选数字,必须挑满picks个,最接近rest但是不超过rest的集合累加和
         */
        int[][][] dp = new int[arr.length + 1][(arr.length + 1) / 2 + 1][sum / 2 + 1];

        /*
        根据base case 初始化dp表
        // base case:没数了,看挑完没,没挑完要返回-1,表示无效
        if (index == arr.length) return picks == 0 ? 0 : -1;
         */
        for (int index = 0; index <= arr.length; index++) {
   
     
            for (int picks = 0; picks <= (arr.length + 1) / 2; picks++) {
   
     
                for (int rest = 0; rest <= sum / 2; rest++) {
   
     
                    dp[index][picks][rest] = -1;
                }
            }
        }
        for (int rest = 0; rest <= sum / 2; rest++) {
   
     
            dp[arr.length][0][rest] = 0;
        }

        /*
        暴力递归外层与内层的依赖关系:
        // index的数不要
        int p1 = process(arr, index + 1, picks, rest);
        // index的数要
        int p2 = -1;
        int next = -1;
        if (arr[index] <= rest) next = process(arr, index + 1, picks - 1, rest - arr[index]);
        if (next != -1) p2 = arr[index] + next;
        // 两个累加和PK一下
        return Math.max(p1, p2);

        index层依赖于index+1层,所以从下往上填
         */
        for (int index = arr.length - 1; index >= 0; index--) {
   
     
            for (int picks = 0; picks <= (arr.length + 1) / 2; picks++) {
   
     
                for (int rest = 0; rest <= sum / 2; rest++) {
   
     
                    // index的数不要
                    int p1 = dp[index + 1][picks][rest];
                    // index的数要
                    int p2 = -1;
                    int next = -1;
                    if (arr[index] <= rest && picks > 0) next = dp[index + 1][picks - 1][rest - arr[index]];
                    if (next != -1) p2 = arr[index] + next;
                    // 两个累加和PK一下
                    dp[index][picks][rest] = Math.max(p1, p2);
                }
            }
        }
        // 根据暴力递归的最外层,确定返回值
        if ((arr.length & 1) == 0) {
   
     
            return dp[0][arr.length / 2][sum / 2];
        } else {
   
     
            return Math.max(dp[0][arr.length / 2][sum / 2], dp[0][arr.length / 2 + 1][sum / 2]);
        }
    }

}

总结

什么题目能用动态规划写

只要一个题目可以使用递归解,而且在递归的过程中,出现了求重复解的情况,就可以优化为动态规划,前提是递归过程的可变参数不会过于复杂(比如数组类型,集合类型)

递归如何优化为动态规划

1、 根据递归的可变参数确定动态规划dp表的维度,一个可变参数,就是一维表,两个可变参数,就是二维表;
2、 根据递归的可变参数的范围,确定dp表的大小;
3、 dp表一个格子代表的含义,就是一层递归的返回值代表的含义;
4、 dp表定好后,根据递归的终止条件(basecase),初始化dp表;
5、 根据外层递归与内层递归的依赖关系,确定dp表的填写顺序;
6、 如果存在枚举行为观察临近格子与当前格子,是否存在重复枚举的部分,如果有,则可以进行枚举行为优化;
7、 根据最外层递归的可变参数的值,确定以dp表中哪个格子作为返回值;

枚举行为优化

存在枚举行为是指递归里面存在for循环嵌套递归来计算本层递归的返回值,或者填dp表某个格子时存在for循环遍历其他格子计算当前格子的值的行为。

如何进行枚举行为的优化?

1、 观察某个格子,计算这个格子的值需要依赖到其他哪些格子;
2、 再观察这个格子的临近的格子(上下左右),计算这个格子的值需要依赖到其他哪些格子;
3、 对比这两个格子,在计算的时候,所依赖的其他格子,是否用重复的部分;
4、 如果有,则可以通过复用临近的这个格子的值(可能要减掉多余的部分),替代掉枚举行为;

空间压缩

比如在填二维dp表的时候,发现填本行时,只依赖到上一行,可以尝试把二维dp表优化为一维表(但不一定可以,要看依赖到哪些格子)

递归的几种尝试模型

1、 从左往右的尝试模型;
比如题目17“较小集合的累加和”,从左往右依次遍历,每一个位置保留和不保留,得出不同结果,这种尝试模型,就是从左往右

2、 一个样本做行,一个样本做列;
比如题目7“求两个字符串的最长公共子序列长度”,两个字符串都是一个样本,要枚举两个字符串不同长度时的情况,得出不同结果,这种尝试模型,就是“一个样本做行,一个样本做列”

3、 范围尝试模型;
比如题目10“最长回文子序列”,在某个范围上进行尝试,通常这种在填dp表时左下半区的dp表是不用填的

4、 业务限制的尝试模型;

什么是参数过于复杂,不能用动态规划?比如N皇后问题

N皇后问题是指在N*N的棋盘上要摆N个皇后,
要求任何两个皇后不同行、不同列,也不在同一条斜线上
给定一个正数n,返回n皇后的摆法有多少种。
n=1,返回1
n=2或3,2皇后和3皇后问题无论怎么摆都不行,返回0
n=8,返回92

/**
 * N皇后问题
 *
 * N皇后问题是指在N*N的棋盘上要摆N个皇后,
 * 要求任何两个皇后不同行、不同列,也不在同一条斜线上
 * 给定一个正数n,返回n皇后的摆法有多少种。
 * n=1,返回1
 * n=2或3,2皇后和3皇后问题无论怎么摆都不行,返回0
 * n=8,返回92
 *
 * 需要用递归,但是不能优化为动态规划的例子
 *
 * Created by huangjunyi on 2022/11/30.
 */
public class DynamicProgramming18 {
   
     

    public static int ways(int n) {
   
     
        if (n < 1) return 0;
        // record[i]表示在第i行的皇后摆在第几列
        int[] record = new int[n];
        return process(n, record, 0);
    }

    private static int process(int n, int[] record, int i) {
   
     
        // 来到第n行,表示全都摆好了,返回一种可行方法数
        if (i == n) return 1;
        int res = 0;
        // 放在第1列,放在第2列...,累加方法数
        for (int j = 0; j < n; j++) {
   
     
            // 摆在该列是否违规,违规则跳过,当前行只与往上的行进行比较,不用管下面的行,下面的行交给下层递归
            boolean flag = true;
            for (int k = 0; k < i; k++) {
   
     
                if (j == record[k] || Math.abs(record[k] - j) == Math.abs(i - k)) {
   
     
                    flag = false;
                    break;
                }
            }
            // 记录:第i行的皇后,摆在j列
            if (flag) {
   
     
                record[i] = j;
                res += process(n, record, i + 1);
            }
        }
        return res;
    }

}

因为可变参数中存在一个record 数组,是无法改为dp表的,或者强行改成dp表会非常的复杂,因为这个record数组有很多种情况,无法像整形那样很简单的确定范围