leetcode 1594. 矩阵的最大非负积 中等

给你一个大小为 m x n 的矩阵 grid 。最初,你位于左上角 (0, 0) ,每一步,你可以在矩阵中 向右向下 移动。

在从左上角 (0, 0) 开始到右下角 (m - 1, n - 1) 结束的所有路径中,找出具有 最大非负积 的路径。路径的积是沿路径访问的单元格中所有整数的乘积。

返回 最大非负积 对**109 + 7** 取余 的结果。如果最大积为 负数 ,则返回-1

**注意,**取余是在得到最大积之后执行的。

示例 1:

复制代码
输入:grid = [[-1,-2,-3],[-2,-3,-3],[-3,-3,-2]]
输出:-1
解释:从 (0, 0) 到 (2, 2) 的路径中无法得到非负积,所以返回 -1 。

示例 2:

复制代码
输入:grid = [[1,-2,1],[1,-2,1],[3,-4,1]]
输出:8
解释:最大非负积对应的路径如图所示 (1 * 1 * -2 * -4 * 1 = 8)

示例 3:

复制代码
输入:grid = [[1,3],[0,-4]]
输出:0
解释:最大非负积对应的路径如图所示 (1 * 0 * -4 = 0)

提示:

  • m == grid.length
  • n == grid[i].length
  • 1 <= m, n <= 15
  • -4 <= grid[i][j] <= 4

分析:由于两个负数相乘可以得到正数,因此除了求最大乘积,还要求最小乘积。令 dp[i][j].mini 代表从左上角(0,0)走到(i,j)的路径最小乘积,dp[i][j].maxn 代表从左上角(0,0)走到(i,j)的路径最大乘积。想要走到点(i,j),要么从上面(i-1,j)走来,要么从左边(i,j-1)走来,可以得到两个计算式:

cpp 复制代码
dp[i][j].mini=min(min(dp[i-1][j].mini*grid[i][j],dp[i-1][j].maxn*grid[i][j]),min(dp[i][j-1].mini*grid[i][j],dp[i][j-1].maxn*grid[i][j]));
dp[i][j].maxn=max(max(dp[i-1][j].mini*grid[i][j],dp[i-1][j].maxn*grid[i][j]),max(dp[i][j-1].mini*grid[i][j],dp[i][j-1].maxn*grid[i][j]));

最后答案就是 dp[m-1][n-1].mini 和 dp[m-1][n-1].maxn 的最大值对 1e9+7 取模的值。注意取余是在得到最大积之后执行的,因此中间值需要用 long long 避免溢出。

cpp 复制代码
long long min(long long a,long long b)
{
    return a>b?b:a;
}
long long max(long long a,long long b)
{
    return a>b?a:b;
}
typedef struct node
{
    long long mini,maxn;
}node;
int maxProductPath(int** grid, int gridSize, int* gridColSize) {
    int m=gridSize,n=*gridColSize;
    long long mod=1e9+7;
    node dp[m+5][n+5];
    dp[0][0].mini=dp[0][0].maxn=grid[0][0]*1LL;
    for(int i=1;i<m;++i)
        dp[i][0].mini=dp[i][0].maxn=1LL*dp[i-1][0].mini*grid[i][0];
    for(int j=1;j<n;++j)
        dp[0][j].mini=dp[0][j].maxn=1LL*dp[0][j-1].mini*grid[0][j];
        
    for(int i=1;i<m;++i)
    {
        for(int j=1;j<n;++j)
        {
            dp[i][j].mini=min(min(dp[i-1][j].mini*grid[i][j],dp[i-1][j].maxn*grid[i][j]),min(dp[i][j-1].mini*grid[i][j],dp[i][j-1].maxn*grid[i][j]));
            dp[i][j].maxn=max(max(dp[i-1][j].mini*grid[i][j],dp[i-1][j].maxn*grid[i][j]),max(dp[i][j-1].mini*grid[i][j],dp[i][j-1].maxn*grid[i][j]));
        }
    }

    return (int)(max(max(dp[m-1][n-1].mini,dp[m-1][n-1].maxn),-1)%mod);
}
相关推荐
6Hzlia7 小时前
【Hot 100 刷题计划】 LeetCode 152. 乘积最大子数组 | C++ 动态规划 (绝妙 swap 翻转技巧)
c++·leetcode·动态规划
smj2302_796826527 小时前
解决leetcode第3901题好子序列查询
python·算法·leetcode
_深海凉_7 小时前
LeetCode热题100-每日温度
算法·leetcode·职场和发展
脱氧核糖核酸__8 小时前
LeetCode热题100——54.螺旋矩阵(题解+答案+要点)
c++·算法·leetcode·矩阵
菜菜的顾清寒8 小时前
力扣hot100(17) 缺失的第一个正数
算法·leetcode·职场和发展
承渊政道8 小时前
【递归、搜索与回溯算法】(二叉树深搜模型拆解与经典题型全面突破)
数据结构·c++·学习·算法·leetcode·macos·bfs
泽02029 小时前
OJBalancer ----- 基于负载均衡仿leetcode的刷题界面
linux·leetcode·负载均衡
_深海凉_19 小时前
LeetCode热题100-最小栈
java·数据结构·leetcode
不知名的忻19 小时前
Morris遍历(力扣第99题)
java·算法·leetcode·morris遍历
_深海凉_20 小时前
LeetCode热题100-除了自身以外数组的乘积
数据结构·算法·leetcode