【动态规划Ⅳ】二维数组的动态规划——最小路径和

news2024/10/7 16:13:00

二维数组的动态规划

  • 最小路径和
    • 64. 最小路径和
      • 原地修改数组
      • 定义二维数组进行状态转移
      • 优化:用 一维数组进行状态转移
      • 相似题目:LCR 166. 珠宝的最高价值
    • 120. 三角形最小路径和
      • 原地修改数组
      • 定义二维数组进行状态转移
      • 一维数组进行状态转移
      • 自底向上,反方向进行状态转移
    • 931. 下降路径最小和
    • 1289. 下降路径最小和 II

这一组题目都非常相似,其实就是遍历数组,不管是长方形的,还是三角形的,遍历的过程中更新状态,并用dp数组记录。 dp数组都可以从二维的优化到一维的。但是需要注意,更新dp[i]的时候,dp[i]依赖的元素,如果是上一轮更新的结果,那么这个元素要在dp[i]之后更新;如果这个元素应该是新一轮更新的结果,那么这个元素要在dp[i]之前更新——这个决定了更新的方向,是从后往前还是从前往后。

最小路径和

64. 最小路径和

64. 最小路径和

给定一个包含非负整数的 m x n 网格 grid ,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。
说明:每次只能向下或者向右移动一步。
在这里插入图片描述

其实这个题还是非常简单的!到一个点只能通过左边和上边两个点,如果用dp这个二维数组进行状态转移的话,dp[i][j]就是到这个点需要的最小路径和,更新过程:dp[i][j] = min( dp[i-1][j], dp[i-1][j])

原地修改数组

直接在原来的数组grid上更新,这样虽然解决了空间复杂度,但是在实际业务中,修改原数组是很危险的(原数组可能在很多地方有其他用处,而且java是面向对象的,传入的数组是引用……)。
java代码如下:

class Solution {
    public int minPathSum(int[][] grid) {
        int row = grid.length; // get number of rows
        int col = grid[0].length; // get number of columns
        // 第一列只能从下网上
        for(int i = 1; i < row; i++)
            grid[i][0] += grid[i-1][0];
        //第一行只能从左往右
        for(int j = 1; j < col; j++)
            grid[0][j] += grid[0][j-1];
        // 更新其他点
        for(int i = 1; i < row; i++)
            for(int j = 1; j < col; j++){
                grid[i][j] += Math.min(grid[i-1][j], grid[i][j-1]);
            }
        return grid[row-1][col-1];
    }
}

定义二维数组进行状态转移

定义一个和原数组一样大小的二维数组,进行状态转移,其实和上面是一样的,只是需要一个新的数组。
java代码如下:

class Solution {
    public int minPathSum(int[][] grid) {
        int row = grid.length; // get number of rows
        int col = grid[0].length; // get number of columns
        int[][] pathCount = new int[row][col];
        pathCount[0][0] = grid[0][0];
        for(int i = 1; i < row; i++)
            pathCount[i][0] = grid[i][0] + pathCount[i-1][0];
        for(int j = 1; j < col; j++)
            pathCount[0][j] =  grid[0][j] + pathCount[0][j-1];        
        for(int i = 1; i < row; i++)
            for(int j = 1; j < col; j++){
                pathCount[i][j] = grid[i][j] + Math.min(pathCount[i-1][j], pathCount[i][j-1]);
            }
        return pathCount[row-1][col-1];
    }
}

优化:用 一维数组进行状态转移

二维状态转移数组,其实每一行只在计算第二行的用到,因此可以之际用一维的,此时更新过程:dp[j] = min(dp[j], dp[j-1]) + grid[i][j]。 遍历的时候一维dp还是从前往后,dp[i]依赖于dp[i-1],而dp[i-1]确实需要先更新。
java代码如下:

class Solution {
    public int minPathSum(int[][] grid) {
        int row = grid.length; // get number of rows
        int col = grid[0].length; // get number of columns
        int[] pathCount = new int[col];
        Arrays.fill(pathCount ,Integer.MAX_VALUE);
        pathCount[0] = 0;        
        for(int i = 0; i < row; i++)
            for(int j = 0; j < col; j++){
                if(j > 0)
                    pathCount[j] = grid[i][j] + Math.min(pathCount[j], pathCount[j-1]);
                else
                pathCount[j] += grid[i][j] ;
            }
        return pathCount[col-1];
    }
}

相似题目:LCR 166. 珠宝的最高价值

LCR 166. 珠宝的最高价值

现有一个记作二维矩阵 frame 的珠宝架,其中 frame[i][j] 为该位置珠宝的价值。拿取珠宝的规则为:

  • 只能从架子的左上角开始拿珠宝
  • 每次可以移动到右侧下侧的相邻位置
  • 到达珠宝架子的右下角时,停止拿取

注意:珠宝的价值都是大于 0 的。除非这个架子上没有任何珠宝,比如 frame = [[0]]。

这题和最小路径和,是一模一样的,只是换了个场景罢了。

class Solution {
    public int jewelleryValue(int[][] frame) {
        int m = frame.length, n = frame[0].length;
        int[][] dp = new int[m][n];
        for (int i = 0; i < m; ++i) {
            for (int j = 0; j < n; ++j) {
                if (i > 0) {
                    dp[i][j] = Math.max(dp[i][j], dp[i - 1][j]);
                }
                if (j > 0) {
                    dp[i][j] = Math.max(dp[i][j], dp[i][j - 1]);
                }
                dp[i][j] += frame[i][j];
            }
        }
        return dp[m - 1][n - 1];
    }
}

120. 三角形最小路径和

120. 三角形最小路径和

给定一个三角形 triangle ,找出自顶向下的最小路径和
每一步只能移动到下一行中相邻的结点上。相邻的结点 在这里指的是 下标 与 上一层结点下标 相同或者等于 上一层结点下标 + 1 的两个结点。也就是说,如果正位于当前行的下标 i ,那么下一步可以移动到下一行的下标 i 或 i + 1

这个题目和上面的题目不同的点在于,这是一个三角形的结构,且一行的状态完全由上一行的节点状态相关。题目中说上一行节点i下次移动到下一行的i或者i+1,那么对于下一行i,其实是由i-1和i中较小值移动来的。
经过上面的分析假设用二维数组dp表示状态转移,那么dp[i][j] = min(dp[i-1][j-1], dp[i-1][j]) + triangle[i][j]

原地修改数组

可以直接在triangle数组上进行状态转移。java代码如下。需要注意的是java中的<List<List<>>>这样的数据结构,之前没咋用,不能直接用下标,需要用set()和get()方法。

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int row = triangle.size(); 
        // 如果只有一行,直接返回
        if(row == 1) 
            return triangle.get(0).get(0);
        for(int i = 1; i < row; i++){
            for(int j = 0; j < i; j++){
            	//第一列只能由上方的点转移来
                if(j==0)  
                    triangle.get(i).set(0, triangle.get(i).get(0) + triangle.get(i-1).get(0));
                // 其他点由其上方和左上方两个中较小的决定
                else
                    triangle.get(i).set(j, triangle.get(i).get(j) + Math.min(triangle.get(i-1).get(j-1), triangle.get(i-1).get(j)));
            }
            //最后一个点只能由左上方的点转移来
            triangle.get(i).set(i, triangle.get(i).get(i) + triangle.get(i-1).get(i-1));
        }
        //最后一行的最小值为结果
        int count = triangle.get(row-1).get(0);
        for(int i =1; i < row; i++)
            count = Math.min(count,triangle.get(row-1).get(i));
        return count;
    }
}

定义二维数组进行状态转移

修改原始数据会有很多潜在威胁,还是定义一个新的二维数组进行状态转移:

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int row = triangle.size(); 
        if(row == 1) 
            return triangle.get(0).get(0);
        // 定义新的数组
        int dp[][] = new int[row][row];
        dp[0][0] = triangle.get(0).get(0);
        for(int i = 1; i < row; i++){
        	//单独处理每行第一个,其只能由上一行第一个转移而来
            dp[i][0] = dp[i-1][0] + triangle.get(i).get(0);
            // 正常处理,由上一行的j-1和j较小的一个转移而来
            for(int j = 1; j < i ; j ++)
                dp[i][j] = Math.min(dp[i-1][j-1], dp[i-1][j]) + triangle.get(i).get(j);
            //每行最后一个只能由上一行最后一个转移而来
            dp[i][i] = dp[i-1][i-1] + triangle.get(i).get(i);
        }
        int count = dp[row-1][0];
        for(int i =1; i < row; i++)
            count = Math.min(count,dp[row-1][i]);
        return count;
    }
}

一维数组进行状态转移

同样的,我们也可以考虑用一维数组进行状态转移,优化空间复杂度,但是需要注意的是,这个题目,更新一维状态数组的时候,需要从后往前! 因为dp[j] = min(dp[j-1], dp[j]) + triangle[i][j]的时候,dp[j-1]应该是上一轮更新的结果,表示上一行的结果;如果从前往后,dp[j-1]以及在这一轮更新过了。因此整体的更新防线:自顶向下;某一层的更新方向(dp状态数组的更新方向)从后往前。
Java代码如下:

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int row = triangle.size(); 
        if(row == 1) 
            return triangle.get(0).get(0);
        int dp[] = new int[row];
        dp[0] = triangle.get(0).get(0);
        for(int i = 1; i < row; i++){
            dp[i] = dp[i-1] + triangle.get(i).get(i);
            for(int j = i -1; j> 0 ; j--)
                dp[j] = Math.min(dp[j-1], dp[j]) + triangle.get(i).get(j);
            dp[0] = dp[0] + triangle.get(i).get(0);
        }

        int count = dp[0];
        for(int i =1; i < row; i++)
            count = Math.min(count,dp[i]);
        return count;
    }
}

自底向上,反方向进行状态转移

题意中,上一层下标为i的,可以传递到下一层下标为i和i+1的。可以考虑将这个传递方向反过来,直接从最后一行开始,从下往上传递,最后到triangle[0][0],得到的就是最终的答案!从下一层到上一层,状态转移为:dp[j] = min(dp[j], dp[j+1]) + triangle[i][j];,也就是说上一层下标为i的元素,由下一层下标为i和i+1的元素影响,这其实就是题意的意思。整体的更先方向是自底向上的, 一行中的更新方向是从前往后的。
Java代码:

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int row = triangle.size(); 
        if(row == 1) 
            return triangle.get(0).get(0);
        int dp[] = new int[row];
        // 将三角形最后一行赋值给dp状态数组
        for(int i = 0; i < row; i++)
            dp[i] = triangle.get(row-1).get(i);
        //从下往上更新
        for(int i = row - 2; i >=0 ; i--){
        	//每一层从前往后更新
            for(int j = 0; j <= i ; j++)
                dp[j] = Math.min(dp[j], dp[j+1]) + triangle.get(i).get(j);
        }
        return dp[0];    
    }
}

931. 下降路径最小和

931. 下降路径最小和

给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的下降路径最小和
下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(即位于正下方或者沿对角线向左或者向右的第一个元素)。具体来说,位置 (row, col) 的下一个元素应当是 (row + 1, col - 1)、(row + 1, col) 或者 (row + 1, col + 1)

根据题意,row行col列的元素,往下可以传递到row+1行的col-1、col以及col+1列,那么对于row+1行的col列,可以从row上的col-1、col以及col+1列传递来,因此定义二维的状态转移数组dp,转移过程:dp[i][j] = min(dp[i-1][j-1], dp[i-1][j], dp[i-1][j+1] + matrix[i][j]。注意边界,第一列和最后一列只能由上一行的两个元素传递来。
java代码如下:

class Solution {
    public int minFallingPathSum(int[][] matrix) {
        int n = matrix.length;
        int[][] dp = new int[n][n];
        // 第一行直接赋值
        for(int i = 0; i < n; i++)
            dp[0][i] = matrix[0][i];
        for(int i=1; i<n; i++){
        	// 第一列的特殊情况
            dp[i][0]= Math.min(dp[i-1][0],dp[i-1][1]) + matrix[i][0];
            for(int j = 1;j < n-1; j++){
                dp[i][j] = matrix[i][j] + Math.min(dp[i-1][j-1], Math.min(dp[i-1][j+1],dp[i-1][j]));
            }
            // 最后一列的特殊情况
            dp[i][n-1] = matrix[i][n-1] + Math.min(dp[i-1][n-1],dp[i-1][n-2]);
        }
        int ans = dp[n-1][0];
        for(int i =1; i<n;i++)
            if(dp[n-1][i]<ans)
                ans = dp[n-1][i];        
        return ans;
    }
}

1289. 下降路径最小和 II

1289. 下降路径最小和 II

给你一个 n x n 整数矩阵 grid ,请你返回 非零偏移下降路径 数字最小值
非零偏移下降路径 定义为:从 grid 数组中的每一行选择一个数字,且按顺序选出来的数字中,相邻数字不在原数组的同一列

这个题目说得天花乱坠的,其实就是row行选了col列,那么row-1行和row=1行都不能选col列,除了col列的其他行都能选择。 这个也可以用二维的dp数组进行状态转移:dp[i][j] = grid[i][j] + min(dp[i-1][k]) k≠j。也就是说,针对一行的每个元素,找到除这个列意外的,dp[i-1]中的最小值即可,代码中可以用三层循环实现:

for(int i = 0; i < row; i++) //遍历行
	for(int j = 0; j < col; j++) // 遍历列
		for(………………)//找dp[i-1]中非j列的min 
		//update dp[i][j]

优化分析:

  • 每次找dp[i-1]行中的最小值除了j列以外的最小值,其实除了最小值对应的minIndex列以外的列,找到的最小值就是dp[i-1]整个行的最小值。而最小值minIndex对应列,要找的最小值,是整个dp[i-1]中的第二个最小值。
  • 最终要求返回的是路径最小和,也就是对最后一行dp[row-1]更新后的最小值。因此我们只需要用一个firstMin变量记录每一行的最小值即可(循环更新的,每一层遍历一行就会更新),最后返回这个值即可。
    最终状态转移:
    在这里插入图片描述
    Java代码如下:
class Solution {
    public int minFallingPathSum(int[][] grid) {
        int row = grid.length;
        int col = grid[0].length; 
        // 记录上一行结果的最小值和第二最小值,以及最小值对应的下标   
        int firstMin = 0; 
        int secondMin = 0; 
        int firstIndex = -1;
        for(int i = 0; i < row; i++){
        	//记录当前行结果的最小值、第二最小值,以及最小值对应的下标
            int curFirMin = Integer.MAX_VALUE;
            int curSecMin = Integer.MAX_VALUE;
            int curFirIndex = -1;
            for(int j = 0; j < col; j++){
            //curSum就是dp[j]
                int curSum = (j != firstIndex ? firstMin : secondMin) + grid[i][j];
                if(curSum < curFirMin){
                    curSecMin = curFirMin;
                    curFirMin = curSum;
                    curFirIndex = j;
                }
                else if(curSum < curSecMin)
                    curSecMin = curSum;
            }
            //用这一轮结果更新上一轮结果
            firstIndex = curFirIndex;
            firstMin = curFirMin;
            secondMin = curSecMin;
        }
        return firstMin; //最终返回firstMin即可
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1917192.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

获取商铺信息,以及商铺信息的增删改查

本文章主要讲述如何对商铺信息进行基本的增删改查操作&#xff0c;及数据库对比。 1、获取首页仪表盘统计数据接口 待收费金额&#xff1a; SELECT count(1) as count,IFNULL(sum(total),0)as sum FROM payment_bill WHERE enabled_mark 1 AND pay_state0 欠费数据&#xf…

Linux-shell编程入门基础

文章目录 前言Shell编程bash特性shell作用域变量环境变量$特殊变量$特殊状态变量 $特殊符号(很重要)其他内置shell命令shell语法的子串截取统计 指令执行时间练习shell特殊扩展变量父子shell的理解内置和外置命令区别 数值计算双括号(())运算letexprexpr模式匹配 bcawk中括号 s…

LabVIEW电容器充放电监测系统

概述 为了对车用超级电容器的特性进行研究&#xff0c;确保其在工作时稳定可靠并有效发挥性能优势&#xff0c;设计了一套车用超级电容器充放电监测系统。该系统通过利用传感器、USB数据采集卡、可调直流稳压电源、电子负载以及信号调理电路&#xff0c;完成对各信号的采集和超…

【matlab】随机森林客户流失预测

目录 引言 核心思想 优点 应用场景 建模步骤 数据集 结果 代码实现 引言 随机森林&#xff08;Random Forest&#xff09;是一种集成学习算法&#xff0c;它通过构建多个决策树并将它们的预测结果进行汇总来提高整体模型的预测准确率、稳定性和泛化能力。随机森林属于“…

CentOS7安装部署git和gitlab

安装Git 在Linux系统中是需要编译源码的&#xff0c;首先下载所需要的依赖&#xff1a; yum install -y curl-devel expat-devel gettext-devel openssl-devel zlib-devel gcc perl-ExtUtils-MakeMaker方法一 下载&#xff1a; wget https://mirrors.edge.kernel.org/pub/s…

软考高项论文备考论

软考高项论文备考是一个系统而全面的过程&#xff0c;需要考生从多个方面进行准备。以下是一份详细的备考策略&#xff0c;供考生参考&#xff1a; 一、明确考试要求 首先&#xff0c;考生需要详细了解软考高项论文的考试要求&#xff0c;包括字数限制、结构要求、评分标准等…

Monaco Editor 中文文档整理(超详细、超全面、带demo示例)

地址 Monaco Editor 中文官网&#xff0c;欢迎大家体验。 网站采用 github pages 部署&#xff0c;可能因网络等原因&#xff0c;响应较慢&#xff0c;请耐心等待~ 也欢迎大家对译文内容进行纠正&#xff0c;如果有误&#xff0c;可本文留言、提 ISSUE、亦或直接在内容页的在 …

苹果手机信号为什么比安卓手机信号差,原因你知道吗?

不知道你有没有过这种体会&#xff0c;在使用苹果手机时&#xff0c;信号要比安卓的信号差的一些&#xff0c;这到底是怎么回事呢&#xff1f;接下来&#xff0c;小编从多个角度深入分析下这一问题&#xff0c;给出相应的解决方案&#xff0c;希望能帮助到大家哈&#xff01; ​…

Mac视频录制神器推荐,让你的创作更高效

“mac可以视频录制吗&#xff1f;最近我开始对录制和编辑视频产生了浓厚的兴趣。然而&#xff0c;由于我主要使用的是mac电脑&#xff0c;我发现关于在mac上录制视频的教程和资源相对较少。大家知道如何在mac上录制视频吗&#xff1f;如果有的话&#xff0c;请告诉我具体方法&a…

[WMCTF2020]easy_re

CTF逆向-[WMCTF2020]easy_re-WP_虚机-perl加载器截取_逆向分析ctf实战 百度网盘-CSDN博客 参考博客 收获 perl,反正这个东西流程和upx壳很像,(高级的SMC?) 它会加载一个脚本,来解密源代码,期间call 一个 script 题解: 运行为第一步! 输入类型 输入类型一般就是加密,(期间…

电商之订单价税拆分实现方案

文章目录 案例数据实现思路1、计算出平均金额2、计算每个商品的金额 实现方案1、订单 order 实体2、订单明细 orderDetail 实体3、实现类4、测试启动5、实现结果 在做电商项目的时候&#xff0c;会遇到要对订单明细进行纳税金额拆分&#xff0c;即将税额拆分到每个商品上&#…

最优化(10):牛顿类、拟牛顿类算法

4.4 牛顿类算法——介绍了经典牛顿法及其收敛性&#xff0c;并介绍了修正牛顿法和非精确牛顿法&#xff1b; 4.5 拟牛顿类算法——引入割线方程&#xff0c;介绍拟牛顿算法以及拟牛顿矩阵更新方式&#xff0c;然后给出了拟牛顿法的全局收敛性&#xff0c;最后介绍了有限内存BFG…

【漏洞复现】宏景HCM-LoadOtherTreeServlet SQL注入

声明&#xff1a;本文档或演示材料仅用于教育和教学目的。如果任何个人或组织利用本文档中的信息进行非法活动&#xff0c;将与本文档的作者或发布者无关。 一、漏洞描述 宏景HCM人力资源信息管理系统是一个全面的人力资源管理软件&#xff0c;它覆盖了人力资源管理的各个模块…

龙蜥Anolis OS基于开源项目制作openssh 9.8p1 rpm包 —— 筑梦之路

环境信息 制作过程和centos 7几乎没有区别&#xff0c;此处就不再赘述。 CentOS 7基于开源项目制作openssh9.8p1 rpm二进制包修复安全漏洞CVE-2024-6387 —— 筑梦之路_cve-2024-6387修复-CSDN博客 制作成果展示 tree RPMS/ RPMS/ └── x86_64├── openssh-9.8p1-1.an7.…

【Python 基础】第一个程序

第一个程序 虽然交互式环境对于一次运行一条 Python 指令很好&#xff0c;但要编写完整的 Python程序&#xff0c;就需要在文件编辑器中输入指令。“文件编辑器”类似于 Notepad 或 TextMate这样的文本编辑器&#xff0c;它有一些针对输入源代码的特殊功能。要在 IDLE 中打开文…

常用知识碎片 Vue3 ref和reactive (内含其他常用知识)

目录 ref和reactive ref reactive 总结&#xff1a; setup语法糖 语法糖是啥&#xff1f; Vue3 setup语法糖 Vue3 不使用setup语法糖示例&#xff1a; Vue3 使用setup语法糖示例&#xff1a; ref和eative主要区别 ref和reactive 在 Vue 3 中&#xff0c;ref 和 reac…

linux radix-tree 基数树实现详解

radix tree&#xff0c;又称做基数树&#xff0c;是一种适合于构建key(index)与value(item)相关联的数据结构。内核中使用非常广泛。本文主要聚焦linux内核基数树的代码实现,大量注释过的代码。 radix-tree组织结构如下: 1、数据结构 /** The bottom two bits of the slot de…

SQL 针对上面的salaries表emp_no字段创建索引idx_emp_no

系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂了就去分享给你的码吧。 描述 针对salaries…

LabVIEW滤波器性能研究

为了研究滤波器的滤波性能&#xff0c;采用LabVIEW设计了一套滤波器性能研究系统。该系统通过LabVIEW中的波形生成函数&#xff0c;输出幅值及频率可调的正弦波和白噪声两种信号&#xff0c;并将白噪声与正弦波叠加&#xff0c;再通过滤波器输出纯净的正弦波信号。系统通过FFT&…

RFID技术革新养猪业,构建智能化养殖场

RFID技术作为无线射频识别技术的一种&#xff0c;凭借着非接触、高效识别的特性&#xff0c;在养殖业行业中得到了广泛的应用&#xff0c;为构建智能化、高效化的养殖场提供了强大的技术支持&#xff0c;给传统养殖业带来了一场前所未有的技术变革。以下是RFID技术在养猪行业不…