算法28:力扣64题,最小路径和------------样本模型

题目:

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

思路:

1. 既然是给定二维数组matrix,那么二维数组的数据是知道的

2. 这个人只能从左上角出发,目的地是右下角

3. 沿途只能向下或者向右走。那么第一行和第一列的值是可以知道的。

假设这个二维数组是:

3796
6539
8315
4792

根据这个二维数组推导:

第一行的数据只能往右走,因为它不存在上方数据,依此可以推导出:

3101925

第一列的数据,只能往下走,因为它不存在左边的数据,以此可以推导

310(3+7)19(10+9)25(19+6)
9(3+6)
17(9+8)
21(17+4)

已经推导出第一行和第一列的数据,接下来开始推导剩下的数据。因为每一个格子的数据都是依赖上一行的当前列或者当前行的前一列的值。谁小,就要谁。以此可以推导出:

3101925
914(9<10取 9+5)17(14<19 取14+3)26(17<25 取17+9)
1717(14<17 取14+3)1823(18<26 取 18+5)
2124 (17<21 取 17+7)27 (18<24 取 18+9)25 (23<27 取 23+2)

推导公式已经出来了。下面看代码的实现:

public static int minSum (int[][] matrix)
    {
        if (matrix == null || matrix.length == 0) {
            return 0;
        }
        //行数
        int row = matrix.length;
        //列数
        int col = matrix[0].length;

        //如果是100*100规模的二维数组。那么dp的空间开销巨大
        int[][] dp = new int[row][col];

        //dp的第一行值只能依赖左边的值
        dp[0][0] = matrix[0][0];
        //构建dp表的第一行数据
        for (int colIndex = 1; colIndex < col; colIndex++) {
            //前一列和当前列的累加和,放入dp表
            dp[0][colIndex] = dp[0][colIndex -1] + matrix[0][colIndex];
        }

        //构建dp表的第一列数据
        for (int rowIndex = 1; rowIndex < row; rowIndex++) {
            //上一列和当前列的累加和,放入dp表
            dp[rowIndex][0] = dp[rowIndex -1][0] + matrix[rowIndex][0];
        }

        for (int rowIndex = 1; rowIndex < row; rowIndex++) {

            for (int colIndex = 1; colIndex < col; colIndex++) {
                //先获取上一行当前列 与 当前行的前一列 较小的值
                //获取matrix数组的当前行当前列的值
                //两者相加,就是 dp[rowIndex][colIndex]能够得到的最小值
                dp[rowIndex][colIndex] = Math.min(dp[rowIndex][colIndex -1], dp[rowIndex -1][colIndex]) + matrix[rowIndex][colIndex];
            }
        }
        return dp[row-1][col-1];
    }

以上代码是逐步推导的过程。但是,如果一个100行100列的数组需要推导,而且是要右下角的数据,中间数据是没有必要一直存在的。以上的代码浪费了 100*100的空间,并不是一个好的方式:

分析:

1. 第一行的数据是一定要的,因为根据第一行数据可以推导出第二行的数据

2. 第一列的数据是没有必要一次性全部推导出来的,因为每一个格子只依赖上一行的当前列和当前行的前一列。如果移动到当前行,直接更新当前行的第一列即可。

优化的代码:

//空间压缩进行优化
    public static int minSum2 (int[][] matrix)
    {
        if (matrix == null || matrix.length == 0) {
            return 0;
        }
        //行数
        int row = matrix.length;
        //列数
        int col = matrix[0].length;

        // 空间压缩。
        // 二维数组变一维数组。
        int[] dp = new int[col];
        //构建dp表的第一行数据
        dp[0] = matrix[0][0];
        for (int colIndex = 1; colIndex < col; colIndex++) {
            //前一列和当前列的累加和,放入dp表
            dp[colIndex] = dp[colIndex -1] + matrix[0][colIndex];
        }

        for (int rowIndex = 1; rowIndex < row; rowIndex++) {
            //当前行第一列数据
            dp[0] += matrix[rowIndex][0];

            for (int colIndex = 1; colIndex < col; colIndex++) {
                 dp[colIndex] = Math.min(dp[colIndex -1], dp[colIndex]) + matrix[rowIndex][colIndex];
            }
        }

        return dp[col -1];
    }

完整代码以及测试结果:

package code03.动态规划_07.lesson4;

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

    public static int minSum (int[][] matrix)
    {
        if (matrix == null || matrix.length == 0) {
            return 0;
        }
        //行数
        int row = matrix.length;
        //列数
        int col = matrix[0].length;

        //如果是100*100规模的二维数组。那么dp的空间开销巨大
        int[][] dp = new int[row][col];

        //dp的第一行值只能依赖左边的值
        dp[0][0] = matrix[0][0];
        //构建dp表的第一行数据
        for (int colIndex = 1; colIndex < col; colIndex++) {
            //前一列和当前列的累加和,放入dp表
            dp[0][colIndex] = dp[0][colIndex -1] + matrix[0][colIndex];
        }

        //构建dp表的第一列数据
        for (int rowIndex = 1; rowIndex < row; rowIndex++) {
            //上一列和当前列的累加和,放入dp表
            dp[rowIndex][0] = dp[rowIndex -1][0] + matrix[rowIndex][0];
        }

        for (int rowIndex = 1; rowIndex < row; rowIndex++) {

            for (int colIndex = 1; colIndex < col; colIndex++) {
                //先获取上一行当前列 与 当前行的前一列 较小的值
                //获取matrix数组的当前行当前列的值
                //两者相加,就是 dp[rowIndex][colIndex]能够得到的最小值
                dp[rowIndex][colIndex] = Math.min(dp[rowIndex][colIndex -1], dp[rowIndex -1][colIndex]) + matrix[rowIndex][colIndex];
            }
        }
        return dp[row-1][col-1];
    }

    //空间压缩进行优化
    public static int minSum2 (int[][] matrix)
    {
        if (matrix == null || matrix.length == 0) {
            return 0;
        }
        //行数
        int row = matrix.length;
        //列数
        int col = matrix[0].length;

        // 空间压缩。
        // 二维数组变一维数组。
        int[] dp = new int[col];
        //构建dp表的第一行数据
        dp[0] = matrix[0][0];
        for (int colIndex = 1; colIndex < col; colIndex++) {
            //前一列和当前列的累加和,放入dp表
            dp[colIndex] = dp[colIndex -1] + matrix[0][colIndex];
        }

        for (int rowIndex = 1; rowIndex < row; rowIndex++) {
            //当前行第一列数据
            dp[0] += matrix[rowIndex][0];

            for (int colIndex = 1; colIndex < col; colIndex++) {
                 dp[colIndex] = Math.min(dp[colIndex -1], dp[colIndex]) + matrix[rowIndex][colIndex];
            }
        }

        return dp[col -1];
    }

    public static void main(String[] args) {
        int[][] arr = {
                {3,7,9,6},
                {6,5,3,9},
                {8,3,1,5},
                {4,7,9,2}
        };

        System.out.println(minSum(arr));
        System.out.println(minSum2(arr));
    }

}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

FreeRTOS——队列及其实战

1.队列概念 1&#xff09;队列是任务到任务、任务到中断、中断到任务数据交流的一种机制&#xff08;消息传递&#xff09; 2&#xff09;队列类似数组&#xff0c;只能存储有限数量、相同类型的数据&#xff0c;在创建时需指定队列长度与队列项大小 3&#xff09;出队入队阻塞…

xshell登录不上虚拟机了

电脑重启后连不上本地虚机了 1、关闭防火墙 2 虚拟机ping得到主机&#xff0c;而主机ping不到虚拟机的解决办法 原因&#xff1a;可能是主机的网络适配器没有调好 首先&#xff0c;找到虚拟机的网络配置器 根据虚拟机的IP信息修改主机虚拟适配器VMnet8 修改ip使得和虚拟机连…

Element-ui自定义input框非空校验

1、vue自定义非空指令&#xff1a; main.js中自定义非空指令 当input框或下拉框中数据更新时&#xff0c;触发校验 Vue.directive(isEmpty,{update:function(el,binding,vnode){if(vnode.componentInstance.value""){el.classList.add("is-required");}e…

[Unity]实时阴影技术方案总结

一&#xff0c;Planar Shadow 原理就是将模型压扁之后绘制在需要接受阴影的物体上&#xff0c;这种方式十分高效&#xff0c;消耗很低。具体实现过程参考Unity Shader - Planar Shadow - 平面阴影。具按照自己的理解&#xff0c;其实就是根据光照方向计算片元在接受阴影的平面…

详解卡尔曼滤波(Kalman Filter)

1. 从维纳滤波到卡尔曼滤波 黑盒&#xff08;Black Box&#xff09;思想最早由维纳&#xff08;Wiener&#xff09;在1939年提出&#xff0c;即假定我们对从数据到估计中间的映射过程一无所知&#xff0c;仅仅用线性估计&#xff08;我们知道在高斯背景下&#xff0c;线性估计…

计算机创新协会冬令营——暴力枚举题目01

首先是欢迎大家参加此次的冬令营&#xff0c;我们协会欢迎所有志同道合的同学们。话不多说&#xff0c;先来看看今天的题目吧。 题目 力扣题号&#xff1a;2351. 第一个出现两次的字母 注&#xff1a;下述题目和示例均来自力扣 题目 给你一个由小写英文字母组成的字符串 s &…

RocketMQ5.0Pop消费模式

前言 RocketMQ 5.0 消费者引入了一种新的消费模式&#xff1a;Pop 消费模式&#xff0c;目的是解决 Push 消费模式的一些痛点。 RocketMQ 4.x 之前&#xff0c;消费模式分为两种&#xff1a; Pull&#xff1a;拉模式&#xff0c;消费者自行拉取消息、上报消费结果Push&#x…

探索Allure Report:提升自动化测试效率的秘密武器

亲爱的小伙伴们&#xff0c;由于微信公众号改版&#xff0c;打乱了发布时间&#xff0c;为了保证大家可以及时收到文章的推送&#xff0c;可以点击上方蓝字关注测试工程师成长之路&#xff0c;并设为星标就可以第一时间收到推送哦&#xff01; 一.使用 Allure2 运行方式-Python…

【操作系统xv6】学习记录4 -CPU上下文:进程上下文、线程上下文、中断上下文

什么是cpu上下文 CPU 寄存器和程序计数器就是 CPU 上下文&#xff0c;因为它们都是 CPU 在运行任何任务前&#xff0c;必须的依赖环境。 什么是 CPU 上下文切换 先把前一个任务的 CPU 上下文&#xff08;也就是 CPU 寄存器和程序计数器&#xff09;保存起来&#xff0c;然后…

equals()比较字符串和MySQL中=比较结果不一致

问题&#xff1a; 普通车辆入园统计结果数量和普通车辆统计列表数量不一致&#xff1f; 列子&#xff1a;数量:967&#xff0c;列表:974 解决问题步骤 对比统计数量和统计列表的统计方法 统计数量代码实现 一&#xff1a;查询出车辆滞留表数据List 二&#xff1a;查询出…

112. 雷达设备(贪心/逆向思考)

题目&#xff1a; 112. 雷达设备 - AcWing题库 输入样例&#xff1a; 3 2 1 2 -3 1 2 1输出样例&#xff1a; 2 思路&#xff1a; 代码&#xff1a; #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #include<…

海外住宅IP代理的工作原理和应用场景分析,新手必看

海外住宅IP代理作为一种技术解决方案&#xff0c;为用户提供了访问全球网络资源和维护隐私安全的方法。本文将介绍海外住宅IP代理的工作原理和应用场景&#xff0c;帮助读者更好地理解和利用这一技术。 一、工作原理 海外住宅IP代理的工作原理基于代理服务器和IP地址的转发。它…

【springboot配置文件加载源码分析】

在Spring Boot的源码中&#xff0c;配置文件的加载是在应用程序启动的早期阶段进行的。具体来说&#xff0c;配置文件加载的主要步骤发生在SpringApplication类的run()方法中的prepareEnvironment方法中&#xff0c;真正读取我们的配置文件还是PropertySourceLoader。 本篇博客…

Docker安装Flarum(开源论坛)

Flarum介绍安装命令 #---------------------------------------------------------- mkdir -p /opt/flarum && cd /opt/flarum #---------------------------------------------------------- docker run -p 8888:8888 --name flarum \ --restartalways \ -v /opt/flar…

靠着这份年终总结,我涨薪8K,成为领导眼中最闪亮的星~

2023 年即将接近尾声&#xff0c;各大公司的“测试媛/猿”们又到了提交年终总结报告的时候了。 每年到这个时候都是抓耳挠腮、冥思苦想的时候&#xff0c;猛然一想&#xff0c;今年跟去年做的事情好像差不多&#xff0c;那么年终总结可以敷衍了事么&#xff1f; 当然是不可以…

chatGPT带你学习设计模式 (二)抽象工厂模式(创建型模式) GURU

深入理解抽象工厂模式 引言 在面向对象编程中&#xff0c;对象的创建是一个常见且关键的挑战。尤其在需要管理一系列相关对象的创建时&#xff0c;传统的对象创建方法&#xff08;如直接使用 new 关键字&#xff09;可能导致代码的高耦合和低灵活性。这时&#xff0c;抽象工厂…

rime中州韵小狼毫 中英互绎 滤镜

英文在日常生活中已经随处可见&#xff0c;我们一般中英互译需要使用专业的翻译软件来实现。但如果我们在输入法中&#xff0c;在输入中文的时候&#xff0c;可以顺便瞟一眼对应的英文词汇&#xff0c;或者在输入英文的时候可以顺便了解对应的中文词汇&#xff0c;那将为我们的…

【Qt第三方库】QXlsx库——对 Excel 文件进行相关操作

0 前言 关键词&#xff1a;Qt&#xff1b;Excel&#xff1b;QXlsx&#xff1b;QInt 简介&#xff1a; QXlsx 是第三方开源的库&#xff0c;能够对 Excel 文件进行相关操作&#xff08;读写等&#xff09; 地址&#xff1a; QXlsx官网 QXlsx的Github主页 1 快速上手 对于第一次…

设置代理IP地址对网络有什么影响?爬虫代理IP主要有哪些作用?

在互联网的广泛应用下&#xff0c;代理IP地址成为了一种常见的网络技术。代理IP地址可以改变用户的上网行为&#xff0c;进而影响网络访问的速度和安全性。本篇文章将探讨设置代理IP地址对网络的影响&#xff0c;以及爬虫代理IP的主要作用。 首先&#xff0c;让我们来了解一下代…

【Java】实验三 抽象类与接口

实验名称 实验三 抽象类与接口 实验目的 1. 深刻理解抽象类、接口的意义。 2. 熟练掌握抽象类和接口的定义、继承抽象类以及实现接口的方法。 3. 理解和掌握多态。 实验内容 &#xff08;一&#xff09;抽象类实验&#xff1a;项目源码中新建一个ahpu.shape的包&a…