力扣动态规划基础版(矩阵型)

news2024/11/24 4:52:56

62.不同路径(唯一路径问题)

62. 不同路径icon-default.png?t=O83Ahttps://leetcode.cn/problems/unique-paths/

方法一:动态规划 

 找状态转移方程,也就是说它从左上角走到右下角,只能往右或者往下走,那么设置一个位置为(i,j)它只能从(i-1,j)或者(i,j-1)走过来那么它的状态转移方程就是除了这个还要去思考边界条件,如果i=0那么就不满足右侧的状态转移方程,同理j

= 0也不行,就是边上的那些数,可以考虑到边上的那些数都可以设置为1,也就是f(i,0)和f(0,j)都是1

class Solution {
    public int uniquePaths(int m, int n) {
    int[][] f = new int [m][n];
    for(int i = 0; i < m; i ++){
        f[i][0] = 1;
    }
    for(int j = 0; j < n ; j++){
        f[0][j] = 1;
    }
    for(int i = 1; i < m; ++i){
        for(int j = 1; j < n; ++j){
            f[i][j] = f[i -1][j]  + f[i][j -1];
        }
    }
    return f[m - 1][n - 1];
    }
}

优化空间复杂度:

这个唯一路径问题可以进行空间复杂度的优化 :由于 f(i,j) 仅与第 i 行和第 i−1 行的状态有关,因此我们可以使用滚动数组代替代码中的二维数组,使空间复杂度降低为 O(n)。

这个思路是。我们不需要存储每一个点,只需要存储它的上面一个点就行,再加上上一列的值即可,从第二行开始,每一行进行遍历只需要原始的值加上上一列的值。边界条件就是第一行都是1(很好想)

class Solution {
    public int uniquePaths(int m, int n) {
        //对于每一行,按列进行遍历
        //第一行先都设为1
        int f[] = new int[n];
        for(int i = 0; i < n; ++i){
            f[i] = 1;
        }
        //从第二行开始,对每一列进行遍历,它的f值只和它的前一列相关
        for(int i = 1; i < m; i++){
            for(int j = 1; j < n; j++){
                f[j]  += f[j - 1];//这个f[j]再每一次行遍历都更新,它就代表着上一次遍历的上一行的内容 
            }
        }
        return f[n - 1];
    }
}

方法二:排列组合

因为它只能向下走或者是向右走,所以就只有

从左上角到右下角的过程中,我们需要移动 m+n−2 次,其中有 m−1 次向下移动,n−1 次向右移动。因此路径的总数,就等于从 m+n−2 次移动中选择 m−1 次向下移动的方案数。实现这个排列组合的式子就行,其实实现这个技巧也是有讲究的,因为上面等于下面两个的和,所以位数是一样的,做一个循环,每一项相乘就行

class Solution {
    public int uniquePaths(int m, int n) {
        long ans = 1;
        for(int x =  n,y = 1; y < m; ++x,++y){
            ans = ans *x/y;
        }
        return (int)ans;
    }
}

64.最小路径和 

64. 最小路径和icon-default.png?t=O83Ahttps://leetcode.cn/problems/minimum-path-sum/还不错这道题自己写出来了,感觉就是前面那个题目的变式,状态转移方程就是第(i,j)个点的最小值由于只能向下或者向右,所以最小值就是它上面的或者左边的加上它自身的值。然后边界条件考虑(0,0),再考虑第一行和第一列。

class Solution {
    public int minPathSum(int[][] grid) {
        int m = grid.length;
        int n = grid[0].length;
        int dp[][] = new int [m][n];
        dp[0][0] = grid[0][0];
        for(int i = 1 ; i < m ;++i){
            dp[i][0] = dp[i -1][0] + grid[i][0];
        }
        for(int j = 1; j < n;++j){
            dp[0][j] = dp[0][j - 1] + grid[0][j];
        }
        for(int i = 1; i < m; i++){
            for(int j = 1; j < n; j++){
                dp[i][j] = grid[i][j] + Math.min(dp[i -1][j],dp[i][j - 1]);
            }
        }
        return dp[m -1][n -1];
    }
}

63.不同路径Ⅱ

63. 不同路径 IIicon-default.png?t=O83Ahttps://leetcode.cn/problems/unique-paths-ii/

这个其实大同小异,但是加了个障碍也就是说带障碍的要做个判断,如果他这是0,这条路就死了直接置为0

然后这次构建dp数组是多了一个空间确实好用 

class Solution {
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int m = obstacleGrid.length;
        int n = obstacleGrid[0].length;
        int dp[][] = new int [m+1][n+1];
        dp[1][1] = 1;
        for(int i = 1; i < m + 1; ++i){
            for(int j = 1;j < n + 1; ++j){
                if(obstacleGrid[i -1][j -1] == 1){
                    dp[i][j] = 0;
                }
                else{
                    dp[i][j] = Math.max(dp[i][j],dp[i - 1][j]+dp[i][j -1]);//这里的max是为了dp[1][1]
                }
            }
        }
        return dp[m][n];
    }
}

做完了这个题我又回头看第62题,确实是这样创建数组更舒服,我就又写了一版62的题解

class Solution {
    public int uniquePaths(int m, int n) {
    int[][] f = new int [m + 1][n + 1];
    f[1][1] = 1;
    for(int i = 1; i < m + 1; ++i){
        for(int j = 1; j < n + 1; ++j){
            f[i][j] = Math.max(f[i -1][j]  + f[i][j -1],f[i][j]);
        }
    }
    return f[m][n];
    }
}

对于63题我觉得不能用这钟方式,因为如果这样设置他的转移方程是求最小值,这样的话边界是0的情况下都算进去了,就错误了。

120.三角形最小路径和

方法一:动态规划 

120. 三角形最小路径和icon-default.png?t=O83Ahttps://leetcode.cn/problems/triangle/看到这个题目有点恍惚了,感觉是和以前的题目没啥区别,状态转移方程也是很好想,就是这一个的最小值就是上面和上面的左边的最小值加上这个值的min,但是三角形边界的情况还是特殊一点,因为就是比如在第i行的第0个发现j-1到达不了,以及在第i行的第i个发现在第i-1行的第i列也到达不了,所以要考虑这个两边的边界情况。其他的部分还是很像的,还需注意的是triangle是数组想取值用get

 for(int i = 1; i < m; ++i){
            dp[i][0] = dp[i -1][0] + triangle.get(i).get(0);
            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);
        }

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int m = triangle.size();
        int [][] dp = new int [m][m];
        dp[0][0] = triangle.get(0).get(0);
        for(int i = 1; i < m; ++i){
            dp[i][0] = dp[i -1][0] + triangle.get(i).get(0);
            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 minTol = dp[m -1][0];
        for(int i = 1; i < m; i++){
            minTol = Math.min(minTol,dp[m -1][i]);
        }
        return minTol;
    }
}

931.下降路径最小和

931. 下降路径最小和icon-default.png?t=O83Ahttps://leetcode.cn/problems/minimum-falling-path-sum/这道题目其实和上一道的思路属于是殊途同归的,需要考虑这些形状的边界的情况,与三角形不同,这个要考虑左右两边,容易出错的点就是在考虑右边的时候注意下标m -1同时考虑最后一排的最小值。做过上面那道题,这道题可以轻松ac

class Solution {
    public int minFallingPathSum(int[][] matrix) {
        int m = matrix.length;
        int dp[][]  = new int [m][m];
        //初始化
        for(int i = 0; i < m; ++i){
            dp[0][i] = matrix[0][i];
        }
        for(int i = 1; i < m; ++i){
            dp[i][0] = matrix[i][0] + Math.min(dp[i - 1][0],dp[i -1][1]); 
            for(int j = 1; j < m - 1; ++j){
                dp[i][j] = matrix[i][j] + Math.min(Math.min(dp[i -1][j -1],dp[i -1][j]),dp[i -1][j + 1]);
            }
            dp[i][m - 1] = matrix[i][m -1] + Math.min(dp[i -1][m -1],dp[i -1][m -2]);
        }
        int minres = dp[m -1][0];
        for(int i = 0; i < m; ++i){
             minres = Math.min(minres,dp[m -1][i]);
        }
        return minres;
    }
}

221.最大正方形 

221. 最大正方形icon-default.png?t=O83Ahttps://leetcode.cn/problems/maximal-square/

方法一:暴力

暴力来说的话还是比较好想的,就是先遍历一遍,如果有1就设置为res为1了,然后就找到这个左上角,去算一下潜在的最大正方形,再去遍历去找,先沿着对角线如果是1的话,就再看看其他地方是否满足,设置flag,如果不满足置为false

class Solution {
    public int maximalSquare(char[][] matrix) {
        int m = matrix.length;int n = matrix[0].length;
        int res = 0;
        if(matrix == null || matrix.length == 0 || matrix[0].length == 0){
            return res;
        }
        for(int i = 0; i < m; ++i){
            for(int j = 0; j < n; ++j){
                if(matrix[i][j] == '1'){
                    res = Math.max(1,res);
                    //计算可能的最大的正方形边长
                    int currentMaxside = Math.min(m - i,n - j);
                    for(int k = 1 ;k < currentMaxside; ++k){
                       //判断新增的一列是否都是1
                       boolean flag = true; 
                       if(matrix[i + k][j + k] == '0'){
                        break;
                       }
                       for(int q = 0; q < k; ++q){
                        if(matrix[i + k][j + q] == '0' || matrix[i + q][j + k] == '0'){
                            flag = false;
                            break;
                        }
                       }
                        if(flag){
                            res = Math.max(res,k + 1);
                        }
                        else{
                            break;
                        }
                    }
                }
            }
        }
        int ress = res * res;
        return ress;
    }
}

方法二:动态规划

这个就有点秒了,考虑的是用 dp(i,j) 表示以 (i,j) 为右下角,且只包含 1 的正方形的边长最大值。

如果该位置的值是 1,则 dp(i,j) 的值由其上方、左方和左上方的三个相邻位置的 dp 值决定。具体而言,当前位置的元素值等于三个相邻位置的元素中的最小值加 1。

这样的话状态转移方程就可以用下面式子来表示

然后注意边界条件就是,如果在第一排或者第一列,瑞国他是1的话,dp[i][j]只能为1 

dp[i][j] = Math.min(Math.min(dp[i -1][j],dp[i - 1][j -1 ]),dp[i][j -1]) + 1
class Solution {
    public int maximalSquare(char[][] matrix) {
        int Maxres = 0;
        int rows = matrix.length;
        int columns = matrix[0].length;
        int [][] dp = new int [rows][columns];
        for(int i = 0; i < rows; ++i){
            for(int j = 0; j < columns; ++j){
                if(matrix[i][j] == '1'){
                    if(i == 0 || j == 0){
                        dp[i][j] = 1;
                    }
                    else{
                        dp[i][j] = Math.min(Math.min(dp[i -1][j],dp[i - 1][j -1 ]),dp[i][j -1]) + 1;
                    }
                Maxres = Math.max(Maxres,dp[i][j]);
                }
            }
        }
        int Maxsquare = Maxres * Maxres;
        return Maxsquare;
    }
}

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

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

相关文章

音视频入门基础:FLV专题(23)——FFmpeg源码中,获取FLV文件音频信息的实现(下)

音视频入门基础&#xff1a;FLV专题系列文章&#xff1a; 音视频入门基础&#xff1a;FLV专题&#xff08;1&#xff09;——FLV官方文档下载 音视频入门基础&#xff1a;FLV专题&#xff08;2&#xff09;——使用FFmpeg命令生成flv文件 音视频入门基础&#xff1a;FLV专题…

A021基于Spring Boot的自习室管理和预约系统设计与实现

&#x1f64a;作者简介&#xff1a;在校研究生&#xff0c;拥有计算机专业的研究生开发团队&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的网站项目。 代码可以查看文章末尾⬇️联系方式获取&#xff0c;记得注明来意哦~&#x1f339; 赠送计算机毕业设计600…

qt QShortcut详解

1、概述 QShortcut是Qt框架中的一个类&#xff0c;它提供了一种创建键盘快捷键的方式。通过QShortcut&#xff0c;开发者可以将特定的键盘组合&#xff08;如CtrlC、AltF4等&#xff09;与应用程序中的动作&#xff08;如复制、关闭窗口等&#xff09;关联起来。当用户在应用程…

注意力机制的目的:理解语义;编码器嵌入高纬空间计算;注意力得分“得到S*V”;解码器掩码和交叉注意力层用于训练;最终的编码器和输出实现大模型

目录 注意力机制的目的:理解语义中的它是小白兔 词编码器嵌入高纬空间 计算注意力得分“得到S*V” 权重QKV:连接权重 训练阶段使用解码器:翻译后的语句 解码器掩码和交叉注意力层用于训练 最终的编码器和输出实现大模型 Transformer模型中,QKV QKV的作用 举例说明…

jmeter常用配置元件介绍总结之取样器

系列文章目录 1.windows、linux安装jmeter及设置中文显示 2.jmeter常用配置元件介绍总结之安装插件 3.jmeter常用配置元件介绍总结之取样器 jmeter常用配置元件介绍总结之取样器 2.取样器2.1.HTTP请求2.2.Debug Sampler2.3.JSR223 Sampler2.4.JDBC Connection Configuration和J…

【大数据学习 | kafka】简述kafka的消费者consumer

1. 消费者的结构 能够在kafka中拉取数据进行消费的组件或者程序都叫做消费者。 这里面要涉及到一个动作叫做拉取。 首先我们要知道kafka这个消息队列主要的功能就是起到缓冲的作用&#xff0c;比如flume采集数据然后交给spark或者flink进行计算分析&#xff0c;但是flume采用的…

从零开始训练一个大语言模型需要多少天?

一&#xff0c;前言 在AI领域&#xff0c;训练一个大型语言模型&#xff08;LLM&#xff09;是一个耗时且复杂的过程。几乎每个做大型语言模型&#xff08;LLM&#xff09;训练的人都会被问到&#xff1a;“从零开始&#xff0c;训练大语言模型需要多久和花多少钱&#xff1f;”…

【SQL50】day 1

目录 1.可回收且低脂的产品 2.寻找用户推荐人 3.使用唯一标识码替换员工ID 4.产品销售分析 I 5.有趣的电影 6.平均售价 7.每位教师所教授的科目种类的数量 8.平均售价 1.可回收且低脂的产品 # Write your MySQL query statement below select product_id from Products w…

Qt菜单功能实现

本文介绍Qt菜单功能实现。 Qt开发过程中&#xff0c;菜单功能用的还是比较多的&#xff0c;本文针对菜单栏和右键菜单功能实现作简要描述。 1.菜单栏 1)界面设计 在界面中添加菜单栏&#xff08;本例中名为“menubar”&#xff09;&#xff0c;并依次添加需要的菜单&#x…

Jupyter Notebook添加kernel的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

Java:二维数组

目录 1. 二维数组的基础格式 1.1 二维数组变量的创建 —— 3种形式 1.2 二维数组的初始化 \1 动态初始化 \2 静态初始化 2. 二维数组的大小 和 内存分配 3. 二维数组的不规则初始化 4. 遍历二维数组 4.1 for循环 ​编辑 4.2 for-each循环 5. 二维数组 与 方法 5.1…

手机内卷下一站,AI Agent

作者 | 辰纹 来源 | 洞见新研社 2024年除夕夜&#xff0c;OPPO在央视春晚即将开始前举办了一场“史上最短发布会”&#xff0c;OPPO首席产品官刘作虎宣布&#xff0c;“OPPO正式进入AI手机时代”。 春节假期刚过&#xff0c;魅族又公开表示&#xff0c;将停止“传统智能手机…

Python实战:调用淘宝API以抓取商品页面数据

在数据驱动的商业决策中&#xff0c;获取电商平台的商品数据至关重要。淘宝作为中国最大的在线购物平台&#xff0c;其商品数据对于市场分析、价格监控和竞品研究等方面都具有极高的价值。本文将通过一个Python实战案例&#xff0c;展示如何调用淘宝API来抓取商品页面的数据。 …

SpringBoot14-任务

任务 14.1异步任务 所谓异步&#xff0c;在某些功能实现时可能要花费一定的时间&#xff0c;但是为了不影响客户端的体验&#xff0c;选择异步执行 案例&#xff1a; 首先创建一个service&#xff1a; Service public class AsyncService {public void hello(){try {Threa…

如何在Android中自定义property

在Android中创建自定义的属性&#xff08;Android property&#xff09;通常用于调试、性能调优或传递应用和系统之间的信息。 以下是如何在Android中创建和使用自定义属性的步骤&#xff1a; 1. 定义属性 在Android中&#xff0c;属性是以“属性名称属性值”形式定义的键值对…

SSH实验5密钥登录Linuxroot用户(免密登录)

当用户尝试通过SSH连接到远程服务器时&#xff0c;客户端会生成一对密钥&#xff1a;公钥和私钥。公钥被发送到远程服务器&#xff0c;并存储在服务器的~/.ssh/authorized_keys文件中。而私钥则由客户端保管&#xff0c;不会传输给服务器。 在连接过程中&#xff0c;客户端使用…

CelebV-Text——从文本生成人脸视频的数据集

概述 近年来&#xff0c;生成模型在根据文本生成和编辑视频方面受到了广泛关注。然而&#xff0c;由于缺乏合适的数据集&#xff0c;生成人脸视频领域仍然是一个挑战。特别是&#xff0c;生成的视频帧质量较低&#xff0c;与输入文本的相关性较弱。在本文中&#xff0c;我们通…

天地图入门|标注|移动飞行|缩放,商用地图替换

“天地图”是国家测绘地理信息局建设的地理信息综合服务网站。集成了来自国家、省、市&#xff08;县&#xff09;各级测绘地理信息部门&#xff0c;以及相关政府部门、企事业单位 、社会团体、公众的地理信息公共服务资源&#xff0c;如果做的项目是政府部门、企事业单位尽量选…

Python、Delphi 和 C++ 复制文件速度比较

比较 Python、Delphi 和 C 在文件处理上的速度&#xff0c;可以分为以下几个方面进行测试和分析&#xff1a;文件读写速度&#xff1a;指的是在这三种语言中执行相同的文件读写操作所花费的时间。文件大小影响&#xff1a;不同语言对小文件和大文件的处理是否有显著不同。并发性…

复现LLM:带你从零认识语言模型

前言 本文会以Qwen2-0.5B模型为例&#xff0c;从使用者的角度&#xff0c;从零开始一步一步的探索语言模型的推理过程。主要内容如下&#xff1a; 从使用的角度来接触模型本地运行的方式来认识模型以文本生成过程来理解模型以内部窥探的方式来解剖模型 1. 模型前台使用 1.1…