IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 「图解大厂面试高频算法题」动态规划-最大正方形 -> 正文阅读

[数据结构与算法]「图解大厂面试高频算法题」动态规划-最大正方形

「图解大厂面试高频算法题」动态规划-最大正方形

题目原链接: https://leetcode-cn.com/problems/maximal-square/

题目介绍

在一个由 ‘0’ 和 ‘1’ 组成的二维矩阵内,找到只包含 ‘1’ 的最大正方形,并返回其面积。

示例1

在这里插入图片描述

输入:matrix = [[“1”,“0”,“1”,“0”,“0”],[“1”,“0”,“1”,“1”,“1”],[“1”,“1”,“1”,“1”,“1”],[“1”,“0”,“0”,“1”,“0”]]
输出:4

示例2

在这里插入图片描述

输入:matrix = [[“0”,“1”],[“1”,“0”]]
输出:1

题目解答

方法一:二维动态规划

思路和算法

我们可以使用动态规划的方法来解决这个问题,创建一个二维数组dp, dp[i][j]表示以matrix[i][j]为右下角,且只包含1的正方形的边长的最大值,如果我们能计算出所有dp的值,那么最大的正方形面积就为max(dp[i][j])的平方了。现在最关键的问题就是我们如何计算出dp[i][j]的值。
在这里插入图片描述

  • 如果matrix[i][j]=0,那很好理解dp[i][j]=0。
  • 如果matrix[i][j]=1,dp[i][j]如何计算呢?请看上图,图中有三个正方形:
    • 蓝色正方形的边长为4,dp数组下标为dp[i-1][j-1]。
    • 绿色正方形的边长为2,dp数组下标为dp[i-1][j]。
    • 黄色正方形的边长为3,dp数组下标为dp[i][j-1]。
      我们从上图中可以看出dp[i][j]的值为3,取决于左上方dp[i-1][j-1],正上方dp[i-1][j],正左方dp[i][j-1]这三个值中最小的一个值再加1,得出状态转移方程为dp[i][j] = min(dp[i-1][j-1], dp[i-1][j],dp[i][j-1])+1。
      那么我们该如何去理解这个表达式呢?可以这样理解:
  • 绿色正方形(dp[i-1][j])表示:可以为以dp[i][j]为右下角的正方形的右边提供边长为2的边。
  • 黄色正方形(dp[i][j-1])表示:可以为以dp[i][j]为右下角的正方形的下边提供边长为3的边。
  • 蓝色正方形(dp[i-1][j-1])表示:可以为以dp[i][j]为右下角的正方形的左边上边提供边长为4的边。
  • 取三者中的最小值(min(dp[i-1][j-1], dp[i-1][j],dp[i][j-1]))是因为根据木桶原理,我们只能取最短的一个边来作为dp[i][j]表示的最大的正方形的边。
  • 加上matrix[i][j]本身的这个正方形的边长1,就得到了以dp[i][j]为右下角最大的正方形的边长。

具体例子讲解

给出一个M=4,N=5的一个matrix。
在这里插入图片描述

  • 绿色表示哨兵,我们不需要对这些dp进行计算。
  • 灰色方格表示当前正在计算的dp[i][j]。
  • 黄色方格是计算dp[i][j]所依赖的dp[i-1][j-1],dp[i-1][j],dp[i][j-1]。

根据状态转移方程我们可以算出dp[3][4] = min(dp[2][3], dp[2][4], dp[3][3]) + 1 = 2,
同理dp[3][5]=2,dp[4][4]=2。
在这里插入图片描述
dp[4][5] = min(dp[4][4], dp[3][4], dp[3][5]) + 1 = 3
在这里插入图片描述

代码实现

class Solution {
    public int maximalSquare(char[][] matrix) {
        if (matrix == null || matrix.length == 0) {
            return 0;
        }
        int M = matrix.length;
        int N = matrix[0].length;
        int[][] dp = new int[M+1][N+1];
        int result = 0;
        for (int i = 1; i <= M; i++) {
            for (int j = 1; j <= N; j++) {
                if (matrix[i-1][j-1] == '0') {
                    continue;
                }
                dp[i][j] = Math.min(dp[i-1][j-1], Math.min(dp[i-1][j], dp[i][j-1])) + 1;
                result = Math.max(result, dp[i][j]);
            }
        }
        return result*result;
    }
}

复杂度分析

时间复杂度:O(MN)
空间复杂度:O(M
N)

方法二:一维动态规划

思路和算法

有一个更优的解法,因为在二维动态规划的实现中dp[i][j]总是以从左到右,从上到下的方向来计算的,所以我们可以对二维数组进行化简变成一维数组,只需要一个prev变量来临时存储dp[i][j-1]。状态转移方程就变为了dp[j] = min(dp[j-1], dp[j], prev) + 1。

  • dp[j-1]表示二维数组中的dp[i-1][j-1]。
  • dp[j]表示二维数组中的dp[i-1][j]。
  • prev 表示二维数组中的dp[i][j-1]。

代码实现

class Solution {
    public int maximalSquare(char[][] matrix) {
        if (matrix == null || matrix.length == 0) {
                return 0;
        }
        int M = matrix.length;
        int N = matrix[0].length;
        int[] dp = new int[N+1];
        int result = 0;
        for (int i = 1; i <= M; i++) {
            int prev = 0;
            for (int j = 1; j <= N; j++) {
                int t = dp[j];
                if (matrix[i-1][j-1] == '0') {
                    dp[j] = 0;
                    continue;
                }
                dp[j] = Math.min(dp[j-1], Math.min(dp[j], prev)) + 1;
                prev = t;
                result = Math.max(result, dp[j]);
            }
        }
        return result*result;
    }
}

复杂度分析

时间复杂度:O(MN)
空间复杂度:O(N)

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-12-11 15:58:41  更:2021-12-11 15:59:16 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 2:23:17-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码