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

分析:由于两个负数相乘可以得到正数,因此除了求最大乘积,还要求最小乘积。令 dpij.mini 代表从左上角(0,0)走到(i,j)的路径最小乘积,dpij.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]));

最后答案就是 dpm-1n-1.mini 和 dpm-1n-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);
}
相关推荐
To_OC7 小时前
LC 1 两数之和:面试第一道必考题,暴力解法直接被面试官 pass
javascript·算法·leetcode
想吃火锅10057 天前
【leetcode】121.买卖股票的最佳时机js/c++
算法·leetcode·职场和发展
凌波粒7 天前
LeetCode--491.递增子序列(回溯算法)
数据结构·算法·leetcode
退休倒计时7 天前
【每日一题】LeetCode 146. LRU 缓存 TypeScript
算法·leetcode·缓存·typescript
小欣加油7 天前
leetcode3612 用特殊操作处理字符串I
数据结构·c++·算法·leetcode·职场和发展
凌波粒7 天前
LeetCode--90.子集II(回溯算法)
数据结构·算法·leetcode
凌波粒7 天前
LeetCode--46.全排列(回溯算法)
数据结构·算法·leetcode
吃着火锅x唱着歌7 天前
LeetCode 2530.执行K次操作后的最大分数
数据结构·算法·leetcode
sheeta19987 天前
LeetCode 每日一题笔记 日期:2026.06.16 题目:3612. 字符串特殊符号处理
笔记·算法·leetcode
CoderYanger7 天前
A.每日一题:2095. 删除链表的中间节点
java·数据结构·程序人生·leetcode·链表·面试·职场和发展