数据结构与算法(九)动态规划

简介: 数据结构与算法(九)动态规划

前言

动态规划和贪心算法类似,但是又有所区别。

贪心算法是由局部最优解,推导出全局最优解。

动态规划也是局部最优解,但是无法通过局部最优解推导全局最优解。

案例

案例1

比如有此题目-来源力扣

若一个字符串中,'L''R' 字符的数量是相同的,那称该字符串为平衡字符串。

请将一个平衡字符串分隔成尽可能多的子字符串,并满足子字符串也是平衡字符串。

如:

输入:s = "RLRRLLRLRL"
输出:4
解释:s 可以分割为 "RL"、"RRLL"、"RL"、"RL" ,每个子字符串中都包含相同数量的 'L' 和 'R'

此时便可以用贪心算法:遍历字符串,遇到一个平衡字符串就直接记录下来,并将后续的字符串当做新的平衡字符串处理

局部最优解,推导出全局最优解

代码

public int balancedStringSplit(String s) {
    int count = 0;
    int cur = 0;
    char[] chars = s.toCharArray();
    for (char c : chars) {
        if(c == 'R'){
            cur++;
        }else {
            cur--;
        }
        if(cur == 0){
            count++;
        }
    }
    return count;
}

案例2

现有一格子大小为5的背包,有三样物品,它们的价值和所占的格子分别为

  • 物品1:价值1500,占格子数1
  • 物品2:价值2000,占格子数3
  • 物品3:价值3000,占格子数4

需要你求将这些物品放入背包的最大价值是多少?

如果此时使用贪心算法解决:

局部最优:背包格子为1格时的最大价值

全局最优:先取1格容量装入物品,剩下的容量当做新背包处理。

此时得出的最大价值为:1500 + 2000 = 3500,而最大价值应为4500

那么此类问题便是要由动态规划方式解决了。

动态规划

动态规划特点:

  • 局部最优解:也就是它会有一个最优子结构
  • 子问题可以重复
  • 状态转移方程:通过把问题分成很多小阶段一段段的转移。从而得出最优解.

动态规范问题的最关键便是:状态转移方程,可以说我们能写出状态转移方程,那动态规划问题基本上就解决了一大半。

案例2为例子

设f(i,n)为背包占格子数为n时放入i物品的最大价值 —— 局部最优解

那么当放入一个占格子数为m物品i时,此时会发生两种情况

  • m <= n 可以放下该物品,f(i,n) = i_value(i物品的价值) + f(i-1, n-m) 剩余背包大小(n-m)放入物品i-1时的最大价值
  • m > n f(i,n) = f(i-1,n)上一个物品(i-1)在背包大小为n时的价值

m <= n还有一种情况,即为f(i-1,n)大于i_value + f(i-1, n-m)

所以总结为:

  • m <= n f(i,n) = Math.max(i_value + f(i-1, n-m), f(i-1,n))
  • m > n f(i,n) = f(i-1,n)

用表格具象的表示:

物品、格子 1 2 3 4 5
物品1 1500 1500 1500 1500 1500
物品2 1500(放不下) 1500(放不下) 2000 3500 3500
物品3 1500(放不下) 1500(放不下) 2000(放不下) 3500(>3000) 4500

编写代码:

public int dp(int[] value, int[] size, int bagSize){
    // 当为物品i,背包n时的最大价值
    int[][] dp = new int[value.length+1][bagSize+1];
    for (int i = 1; i <= value.length; i++) {
        // 遍历背包的格子,计算当格子大小为n时的最大价值
        for (int n = 1; n <= bagSize; n++) {
            // 当前物品所占格子数
            int m = size[i-1];
            if(m <= n){
                // 放得下
                // f(i,n) = Math.max(i_value + f(n-m), f(i-1,n))
                dp[i][n] = Math.max(value[i-1] + dp[i-1][n-m], dp[i-1][n]);
            }else {
                // f(i,n) = f(i-1,n)
                dp[i][n] = dp[i-1][n];
            }
        }
    }
    return dp[dp.length-1][dp[0].length-1];
}
public static void main(String[] args) {
    System.out.println(dp(new int[]{1500, 2000, 3000}, new int[]{1,3,4}, 4));
    System.out.println(dp(new int[]{1500, 2000, 3000}, new int[]{1,3,4}, 5));
    System.out.println(dp(new int[]{10, 20, 50}, new int[]{1,2,4}, 5));
    System.out.println(dp(new int[]{6, 10, 12}, new int[]{1,2,4}, 5));
}
目录
相关文章
|
3月前
|
机器学习/深度学习 存储 算法
动态规划算法深度解析:0-1背包问题
0-1背包问题是经典的组合优化问题,目标是在给定物品重量和价值及背包容量限制下,选取物品使得总价值最大化且每个物品仅能被选一次。该问题通常采用动态规划方法解决,通过构建二维状态表dp[i][j]记录前i个物品在容量j时的最大价值,利用状态转移方程避免重复计算子问题,从而高效求解最优解。
481 1
|
存储 算法
深入了解动态规划算法
深入了解动态规划算法
336 1
|
10月前
|
存储 算法 Java
算法系列之动态规划
动态规划(Dynamic Programming,简称DP)是一种用于解决复杂问题的算法设计技术。它通过将问题分解为更小的子问题,并存储这些子问题的解来避免重复计算,从而提高算法的效率。
380 4
算法系列之动态规划
|
算法 测试技术 C++
【动态规划算法】蓝桥杯填充问题(C/C++)
【动态规划算法】蓝桥杯填充问题(C/C++)
|
算法 开发者 Python
惊呆了!Python算法设计与分析,分治法、贪心、动态规划...这些你都会了吗?不会?那还不快来学!
【7月更文挑战第10天】探索编程巅峰,算法至关重要。Python以其易读性成为学习算法的首选。分治法,如归并排序,将大问题拆解;贪心算法,如找零问题,每步求局部最优;动态规划,如斐波那契数列,利用子问题解。通过示例代码,理解并掌握这些算法,提升编程技能,面对挑战更加从容。动手实践,体验算法的神奇力量吧!
201 8
|
11月前
|
算法 Java C++
【潜意识Java】蓝桥杯算法有关的动态规划求解背包问题
本文介绍了经典的0/1背包问题及其动态规划解法。
339 5
|
10月前
|
算法 安全 调度
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
|
10月前
|
机器学习/深度学习 算法 测试技术
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
|
算法
动态规划算法学习三:0-1背包问题
这篇文章是关于0-1背包问题的动态规划算法详解,包括问题描述、解决步骤、最优子结构性质、状态表示和递推方程、算法设计与分析、计算最优值、算法实现以及对算法缺点的思考。
593 2
动态规划算法学习三:0-1背包问题
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
259 2

热门文章

最新文章