P1387 最大正方形 题解

这篇具有很好参考价值的文章主要介绍了P1387 最大正方形 题解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

方法1 二分+暴力+前缀和Check

注意细节

通过二维前缀和判定矩形内是否全为1,计算和等于长度的平方就判断为是

复杂度\(\Theta (n^2\log{n})\)

#include <bits/stdc++.h>

#define N (int)(105)

using namespace std;

int mp[N][N];
int s[N][N];
int n,m;

bool check(int lenth)
{
	for(int i = 1;i + lenth - 1 <= n;i++)
	{
		for(int j = 1;j + lenth - 1 <= m;j++)
		{
			if(s[i + lenth - 1][j + lenth - 1] - s[i-1][j + lenth - 1] - s[i + lenth - 1][j-1] + s[i-1][j-1] == lenth * lenth)//
			{
				return true;
			}
		}
	}
	return false;
}

int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);cout.tie(0);
	cin >> n >> m;
	for(int i = 1;i <= n;i++)
	{
		for(int j =1;j <= m;j++)
		{
			cin >> mp[i][j];
			s[i][j] = s[i-1][j] + s[i][j-1] - s[i-1][j-1] + mp[i][j];
		}
	}
	
	int l = 1, r = min(n,m);
	while(l < r)
	{
		int mid = (l + r) / 2 + 1;
		if(check(mid)) l = mid;
		else r = mid - 1;
	}
	cout << l;
	return 0;
}

方法2 DP

设状态\(f_{i,j}\)为以第\(i\)\(j\)列为右下角的最大正方形的边长,\(a_{i,j}\)表示输入矩阵中的数值,有转移方程:

\[f_{i,j} = (min(f_{i-1,j},f_{i,j-1},f_{i-1,j-1}) + 1) * a_{i,j} \]

解释:考虑\(a_{i,j}\)为0,那么\(f_{i,j}\)为零是正确的。

\(a_{i,j}\)不为0,那么最少边长就是1,考虑像上向右延伸边,由于正方形的相关性质,取可以延伸的宽和高的最小,可以延伸的高就是\(min(f_{i-1,j},f_{i-1,j-1})\), 可延伸的宽就是\(min(f_{i,j-1},f_{i-1,j-1})\)。这样就解释完了。

不过我们可以逆向思考一下如何想出这种DP,首先我们是拿一块已经确定了的正方形然后考虑如何将它拓展(刷表思路);或者我们思考如何将其他正方形的交拼成一个新的正方形(填表思路)。

我们画出一幅由1(或者0)组成的地图,并在其中寻找正方形。(建议画图)

刷表思路:我们考虑将一个正方形扩展到它右下一格的位置,那么我们发现,根据状态的定义,右下正方形最大边长,就是1加上当前最大正方形的边长,更长的边长是不支持的。然后我们发现还需要两个条件,就是下侧和右侧两个“条”状部位需要全部都是1,那么这里就可以推出\(min\)的使用了。可见思考\(min\)可以先固定化一部分,再想另一部分。

填表思路:我们考虑以一个位置为右下角、某个固定大小的正方形,如何用三个正方形把它拼出来(取交集),实际上我们发现,我们把右下角刨去之后,观察剩下的部分,需要\(f_{i-1,j},f_{i,j-1},f_{i-1,j-1}\)综合判断,不能缺在边上(\(f_{i-1,j},f_{i,j-1},\)),也不能缺在左上角(\(f_{i-1,j-1}\)),对于这三个量的贡献,我们固定其他两个,看其中一个,都能够发现\(min\)的贡献关系。文章来源地址https://www.toymoban.com/news/detail-599104.html

总结:设置状态的时候占右下角,是为了带最优性质和推理基础;转移的时候从具体例子考虑。

#include <bits/stdc++.h>

#define N (int)(105)

using namespace std;

int mp[N][N];
int f[N][N];
int n,m,ans;

int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);cout.tie(0);
	cin >> n >> m;
	for(int i = 1;i <= n;i++)
	{
		for(int j = 1;j <= m;j++)
		{
			cin >> mp[i][j];
		}
	}
	for(int i = 1;i <= n;i++)
	{
		for(int j = 1;j <= m;j++)
		{
			f[i][j] = (min(min(f[i-1][j-1],f[i-1][j]),f[i][j-1]) + 1) * mp[i][j];
			ans = max(ans,f[i][j]);
		}
	}
	cout << ans;
	
	return 0;
}

到了这里,关于P1387 最大正方形 题解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 力扣221.最大正方形(动态规划)

    思路: 思路:从[0,0]元素开始,计算每个元素对应其与[0,0]之间矩阵块中最大正方形边长 情况:1)matrix [ i , j ] = ‘0’ -- 元素对应的最大正方形为0。 情况:2)matrix [ i , j ] = ‘1’ -- max ( matrix [ i-1 , j ] , matrix [ i - 1 , j - 1 ) ,matrix [ i , j - 1 ] ) + 1 具体实现:1)先找出第一行和第

    2024年02月13日
    浏览(25)
  • 【LeetCode-中等】221. 最大正方形(详解)

    在一个由  \\\'0\\\'  和  \\\'1\\\'  组成的二维矩阵内,找到只包含  \\\'1\\\'  的最大正方形,并返回其面积。 力扣原题链接   暴力法一般不是最优解,但是可以拿来练手 由于正方形的面积等于边长的平方,因此要找到最大正方形的面积,首先需要找到最大正方形的边长,然后计算最大边

    2024年02月13日
    浏览(38)
  • 最大正方形(力扣)暴力 + 动态规划 JAVA

    在一个由 ‘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: 输入:m

    2024年02月15日
    浏览(51)
  • LeetCode221.Maximal-Square<最大正方形>

        这题是动态规划,但是不会写。想着判断dp的 上,左,左上  去了。发现只能这样最大只能判断面积为4的正方形因为只会判断另外三个方格。而要想判断更大的正方形那就需要递归操作了。那肯定会超时了。 好吧,只能看答案了。 正方形的面积的长乘宽。在例子中我们

    2024年02月15日
    浏览(26)
  • 【LeetCode热题100】打卡第39天:数组中第K个最大元素&最大正方形

    大家好,我是知识汲取者,欢迎来到我的LeetCode热题100刷题专栏! 精选 100 道力扣(LeetCode)上最热门的题目,适合初识算法与数据结构的新手和想要在短时间内高效提升的人,熟练掌握这 100 道题,你就已经具备了在代码世界通行的基本能力。在此专栏中,我们将会涵盖各种

    2024年02月16日
    浏览(32)
  • Golang每日一练(leetDay0079) 最大正方形、完全二叉树节点数

    目录 221. 最大正方形 Maximal Square  🌟🌟 222. 完全二叉树的节点个数 Count Complete Tree Nodes  🌟🌟 🌟 每日一练刷题专栏 🌟 Rust每日一练 专栏 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 在一个由  \\\'0\\\'  和  \\\'1\\\'  组成的二维矩阵内,找到只包

    2024年02月08日
    浏览(28)
  • Python-彩色正方形

    核心代码 核心代码 核心代码

    2024年02月19日
    浏览(27)
  • python绘制螺旋正方形

    1.首先导入画图功能库turtle 2.设置画笔大小,颜色  turtle.pensize()  turtle.pencolor() 3.如何绘制:螺旋正方形一开始外边有3条长度相同的边,先勾勒出一条边循环3次将外边构出,里边每勾勒一条循环两次,再进行循环,即需要一个双循环,边每次比前一次缩短。 画笔前进  turt

    2024年04月15日
    浏览(31)
  • CSS 3D旋转正方形

    2024年01月23日
    浏览(28)
  • 将正方形矩阵顺时针转动 90°

    【题目】 给定一个 N×N 的矩阵 matrix,把这个矩阵调整成顺时针转动 90°后的形式。 例如: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 顺时针转动 90°后为: 13 9 5 1 14 10 6 2 15 11 7 3 16 12 8 4 【要求】 额外空间复杂度为 O(1)。 思路: 这里使用分圈处理的方式,在矩阵中用左上角的坐标(tR,tC)和

    2024年02月13日
    浏览(24)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包