可视化图解算法60: 矩阵最长递增路径

牛客网 面试笔试 TOP101

1. 题目

描述

给定一个 n 行 m 列矩阵 matrix ,矩阵内所有数均为非负整数。 你需要在矩阵中找到一条最长路径,使这条路径上的元素是递增的。并输出这条最长路径的长度。

这个路径必须满足以下条件:

  1. 对于每个单元格,你可以往上,下,左,右四个方向移动。 你不能在对角线方向上移动或移动到边界外。

  2. 你不能走重复的单元格。即每个格子最多只能走一次。

数据范围:1≤n ,m≤1000,0 ≤matrix [i] [j]≤1000

进阶:空间复杂度 O(nm) ,时间复杂度 O(nm)

例如:当输入为[[1,2,3],[4,5,6],[7,8,9]]时,对应的输出为5,

其中的一条最长递增路径如下图所示:

示例1

输入:

\[1,2,3\],\[4,5,6\],\[7,8,9\]

返回值:

5

说明:

1->2->3->6->9即可。当然这种递增路径不是唯一的。

示例2

输入:

\[1,2\],\[4,3\]

返回值:

4

说明:

1->2->3->4

备注:

矩阵的长和宽均不大于1000,矩阵内每个数不大于1000

2. 解题思路

首先,我们需要明确题目的要求:

对应的思路如下:

如果文字描述的不太清楚,你可以参考视频的详细讲解。

3. 编码实现

核心代码如下:

Go 复制代码
/**
 * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
 *
 * 递增路径的最大长度
 * @param matrix int整型二维数组 描述矩阵的每个数
 * @return int整型
 */
func solve(matrix [][]int) int {
	// write code here
	maxMat = newArray(len(matrix), len(matrix[0]))
	maxValue := 0 //最长路径
	for i := 0; i < len(matrix); i++ {
		for j := 0; j < len(matrix[i]); j++ {
			//二维数组中的每一个点都可能是最大路径的起始点
			preMax := dfs(matrix, i, j, -1)
			maxValue = max(maxValue, preMax)
		}
	}
	return maxValue
}

var (
	// 如果我们已经知道以该点为头的最长递增路径长度,那么在dfs查找时可以直接使用这个长度,而无需再次计算.
	// 用一个矩阵将已经计算得到的最长递增路径进行存储
	maxMat [][]int
)

// 以坐标(i,j)为起始点的最长路径
func dfs(mat [][]int, i int, j int, pre int) int {
	//2.递归终止条件:不是递增,直接返回
	if mat[i][j] <= pre {
		return 0
	}

	//3.剪枝:如果该点已经计算过,直接返回,不用再重新计算
	if maxMat[i][j] != 0 {
		return maxMat[i][j]
	}

	maxVal := 0
	// 1. 递归步骤
	//1.1 向左
	if i > 0 {
		//用 mat[i][j] 作为 pre,去比较其左侧的数据
		perMax := dfs(mat, i-1, j, mat[i][j])
		maxVal = max(maxVal, perMax)
	}
	//1.2 向右
	if i < len(mat)-1 {
		//用 mat[i][j] 作为 pre,去比较其右侧的数据
		perMax := dfs(mat, i+1, j, mat[i][j])
		maxVal = max(maxVal, perMax)
	}
	//1.3 向上
	if j > 0 {
		//用 mat[i][j] 作为 pre,去比较其上侧的数据
		perMax := dfs(mat, i, j-1, mat[i][j])
		maxVal = max(maxVal, perMax)
	}
	//1.4 向下
	if j < len(mat[i])-1 {
		//用 mat[i][j] 作为 pre,去比较其下侧的数据
		perMax := dfs(mat, i, j+1, mat[i][j])
		maxVal = max(maxVal, perMax)
	}
	maxMat[i][j] = maxVal + 1 //最长路径:上下左右最长的路径+1(当前的点)
	return maxMat[i][j]
}
func max(a, b int) int {
	if a >= b {
		return a
	}
	return b
}

func newArray(row, column int) [][]int {
	arr := make([][]int, row)
	for i := 0; i < row; i++ {
		arr[i] = make([]int, column)
	}
	return arr

}

具体完整代码你可以参考下面视频的详细讲解。

4.小结

矩阵的最长递增路径通过遍历+递归的思想完成。对二维数组中的每一个位置查找最长递增路径。对于每一个点(i,j)来说,分别向上、向下、向左、向右寻找最长递增路径。为了减少递归调用的次数,用一个二维数组maxMat来保留每个点对应的最长递增路径。

《数据结构与算法》深度精讲课程正式上线啦!7 大核心算法模块全解析:

✅ 链表

✅ 二叉树

✅ 二分查找、排序

✅ 堆、栈、队列

✅ 回溯算法

✅ 哈希算法

✅ 动态规划

无论你是备战笔试面试 、提升代码效率 ,还是突破技术瓶颈,这套课程都将为你构建扎实的算法思维底座。🔥立即加入学习打卡,与千名开发者共同进阶!

对于LeetCode数据结构与算法,我们总结了一套【可视化+图解】方法,依据此方法来解决相关问题,算法变得易于理解,写出来的代码可读性高也不容易出错。具体也可以参考视频详细讲解。

今日佳句:奇文共欣赏,疑义相与析。

相关推荐
董董灿是个攻城狮8 小时前
AI视觉连载8:传统 CV 之边缘检测
算法
AI软著研究员15 小时前
程序员必看:软著不是“面子工程”,是代码的“法律保险”
算法
FunnySaltyFish15 小时前
什么?Compose 把 GapBuffer 换成了 LinkBuffer?
算法·kotlin·android jetpack
颜酱16 小时前
理解二叉树最近公共祖先(LCA):从基础到变种解析
javascript·后端·算法
地平线开发者1 天前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮1 天前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者1 天前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考1 天前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习
HXhlx1 天前
CART决策树基本原理
算法·机器学习
Wect2 天前
LeetCode 210. 课程表 II 题解:Kahn算法+DFS 双解法精讲
前端·算法·typescript