来自北大算法课的Leetcode题解:85. 最大矩形

本文探讨了一种使用动态规划解决Leetcode中找到二维矩阵内最大矩形面积的问题。作者首先尝试了错误的dp策略,然后逐步优化,最终实现了一个嵌套循环的动态规划解决方案,通过维护dp_top和dp_left数组来跟踪最大高度和宽度,避免陷入局部最优解。文章还讨论了错误思路及其改进过程。
摘要由CSDN通过智能技术生成

代码仓库Github | Leetcode solutions @doubleZ0108 from Peking University.

  • 解法1(T13% S88%): 动态规划(嵌套一层循环的dp)。思想其实并不复杂,每一个位置看看它能达到的向上最大高度和向左最大宽度,这可以通过一个dp来求,假设获取了这个信息,只需在这个点往左的宽度内依次遍历,每个位置向上取最小高度,并依次计算面积,最终保留最大值即可

    而dp是如何做的呢?可以用两个dp数组分别来做(实测比把两个数组变为一个消耗内存少)

    dp_top[i][j] = dp_top[i-1][j] + 1
    dp_left[i][j] = dp_left[i][j-1] + 1
    

之前错误的想法也记录一下:

最开始想的dp[i][j]=[x,y]代表以i,j为右下角的最大矩形,它的高为x,宽为y,然后依次从正上方(i-1,j),正左方(i,j-1),和对角线(i-1,j-1)进行扩展,但会陷入局部最优解,例如这个例子:

0 0 1 0 
0 0 1 0 
0 0 1 0 
0 0 1 1 
0 1 1 1 
0 1 1 1 
1 1 1 1

因为只能从旁边的一个位置扩展,最终会陷入右下角24的矩形,而不是33的最大结果中

于是又进行优化,把上面的情况在三个方向分别做,开了三个dp数组top、left、corner,把从每个方向的三种扩展都记录下来,这样就可以防止掉入一个贪婪的局部最大值,但会遇到这个样例:

1 1 1 1 1 1 1 1 
1 1 1 1 1 1 1 0 
1 1 1 1 1 1 1 0 
1 1 1 1 1 0 0 0 
0 1 1 1 1 0 0 0

最终会掉入从00开始的5*5矩形里

于是又在三种扩展中加了向左和向上的循环判断这条边必须全为1

总之最后还是会没法很好的扩展

究其根本,我觉得是没有想到dp也是可以跟循环嵌套着来的,不能觉得循环耗时,一重循环还是很轻量的

class Solution(object):
    def maximalRectangle(self, matrix):
        """
        :type matrix: List[List[str]]
        :rtype: int
        """
        m, n = len(matrix), len(matrix[0])

        dp_top = [[0]*(n+1) for _ in range(m+1)]
        dp_left = [[0]*(n+1) for _ in range(m+1)]

        maxArea = 0
        for i in range(1,m+1):
            for j in range(1,n+1):
                if matrix[i-1][j-1] == "1":
                    dp_top[i][j] = dp_top[i-1][j]+1
                    dp_left[i][j] = dp_left[i][j-1]+1

                    minHeight = dp_top[i][j]
                    for k in range(j, j-dp_left[i][j], -1):
                        width = j-k+1
                        minHeight = min(minHeight, dp_top[i][k])
                        maxArea = max(maxArea, minHeight*width)

        return maxArea

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

doubleZ0108

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值