NC20471 [ZJOI2007]棋盘制作
题目
题目描述
国际象棋是世界上最古老的博弈游戏之一,和中国的围棋、象棋以及日本的将棋同享盛名。
据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳。
而我们的主人公小Q,正是国际象棋的狂热爱好者。作为一个顶尖高手,他已不满足于普通的棋盘与规则,于是他跟他的好朋友小W决定将棋盘扩大以适应他们的新规则。
小Q找到了一张由N*M个正方形的格子组成的矩形纸片,每个格子被涂有黑白两种颜色之一。小Q想在这种纸中裁减一部分作为新棋盘,当然,他希望这个棋盘尽可能的大。
不过小Q还没有决定是找一个正方形的棋盘还是一个矩形的棋盘(当然,不管哪种,棋盘必须都黑白相间,即相邻的格子不同色),所以他希望可以找到最大的正方形棋盘面积和最大的矩形棋盘面积,从而决定哪个更好一些。于是小Q找到了即将参加全国信息学竞赛的你,你能帮助他么?
输入描述
第一行包含两个整数N和M,分别表示矩形纸片的长和宽。
接下来的N行包含一个N * M的01矩阵,表示这张矩形纸片的颜色(0表示白色,1表示黑色)。
输出描述
包含两行,每行包含一个整数。
第一行为可以找到的最大正方形棋盘的面积,
第二行为可以找到的最大矩形棋盘的面积(注意正方形和矩形是可以相交或者包含的)。
示例1
输入
3 3
1 0 1
0 1 0
1 0 0
输出
4
6
备注
对于20%的数据,N, M ≤ 80
对于40%的数据,N, M ≤ 400
对于100%的数据,N, M ≤ 2000
题解
方法一
知识点:线性dp,悬线法。
悬线法:
每个点都有独立左右延长的最大宽度。
每个点都有一个往上延长的最大高度,但最大宽度会被其他点的最大宽度限制。也就是说,这个点的高度经过的所有点的最大宽度的最小值是实际这个点的最大宽度。
对一个点先延长其高度,在取高度经过点的最大宽度的最小值,可以得到这个点的高度最大的最大矩形。
由于最大矩形一定是某个点的最大高度,及其高度经过点的最大宽度的最小值构成,所以通过这个方法一定能找到。证明如下:
假设矩形的所有底部点的最大高度比这个矩形的高度都大,那么显然我们可以把这个矩形的高度往上延长直到这些点最大高度的最小值而不会改变宽度,可以使得矩形更大,因此最大矩形的高度一定是某个点的最大高度。那么在一个高度限制下,一定是宽度最大的矩形更大。综上,对所有点优先延长其高度,然后再是左右扩展。
要注意的是,高度,左延长,右延长,三个量一个都不能少。少了高度算不出,少了左右的一个会少算矩形,因为没有左右的传递性,左侧的最大矩形,并不一定是右侧的最大矩形,遍历到右边也许不是同一个矩形了,所以都要算。
这道题是悬线法的运用。
先预处理出每个点的左右延展的最远位置,初始化高度为 \(1\) 。
然后开始向下遍历,逐行更新。可以看到在这个规则下,也是可以满足悬线法要求的,即 1,2,3的要求(宽度不会随高度变化)。
具体细节看代码更清楚。
时间复杂度 \(O(nm)\)
空间复杂度 \(O(nm)\)
方法二
知识点:线性dp,单调栈。
悬线法能解决的问题是单调栈的子集。
这道题也能用单调栈做,只要以行为底线遍历计算最大矩形即可,先处理一行的每个点的最大高度,然后和单调栈那道直方图的题一样处理。
当然要注意每行可能有若干条底线。
时间复杂度 \(O(nm)\)
空间复杂度 \(O(nm)\)
代码
方法一
#include <bits/stdc++.h>
using namespace std;
bool dt[2007][2007];
int l[2007][2007], r[2007][2007], u[2007][2007];///有且仅有三个即可
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n, m;
cin >> n >> m;
for (int i = 1;i <= n;i++)
for (int j = 1;j <= m;j++)
cin >> dt[i][j], l[i][j] = r[i][j] = j, u[i][j] = 1;
for (int i = 1;i <= n;i++)///水平左最长
for (int j = 2;j <= m;j++)
if (dt[i][j - 1] != dt[i][j]) l[i][j] = l[i][j - 1];
for (int i = 1;i <= n;i++)///水平右最长
for (int j = m - 1;j >= 1;j--)
if (dt[i][j] != dt[i][j + 1]) r[i][j] = r[i][j + 1];
int ans1 = 0, ans2 = 0;
for (int i = 1;i <= n;i++) {
for (int j = 1;j <= m;j++) {
if (i > 1 && dt[i - 1][j] != dt[i][j]) {///高度优先的向下传递更新
u[i][j] = u[i - 1][j] + 1;
l[i][j] = max(l[i][j], l[i - 1][j]);
r[i][j] = min(r[i][j], r[i - 1][j]);
}
int a = r[i][j] - l[i][j] + 1;
ans1 = max(ans1, min(a, u[i][j]) * min(a, u[i][j]));
ans2 = max(ans2, a * u[i][j]);
}
}
cout << ans1 << '\n' << ans2 << '\n';
return 0;
}
方法二
#include <bits/stdc++.h>
using namespace std;
bool dt[2007][2007];
int l[2007], r[2007], u[2007];
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n, m;
cin >> n >> m;
for (int i = 1;i <= n;i++)
for (int j = 1;j <= m;j++)
cin >> dt[i][j];
int ans1 = 0, ans2 = 0;
for (int i = 1;i <= n;i++) {
for (int j = 1;j <= m;j++) {
if (i > 1 && dt[i - 1][j] != dt[i][j]) u[j]++;
else u[j] = 1;
}
for (int j = 1;j <= m;j++) {
stack<int> s;
int k;
///不必再考虑上一层同列的扩展长度取最小值
///因为如果这层底部能扩展,则说明底部元素不同,则同一高度元素一定不同
///因此长度和高度是相匹配的,不需要考虑上一次长度
for (k = j;k <= m && (k == j || dt[i][k - 1] != dt[i][k]);k++) {
while (!s.empty() && u[k] <= u[s.top()]) s.pop();
l[k] = s.empty() ? j : s.top() + 1;///是有效位置前一个的位置,记得+1
s.push(k);
}
j = k - 1;
}
for (int j = m;j >= 1;j--) {
stack<int> s;
int k;
for (k = j;k >= 1 && (k == j || dt[i][k] != dt[i][k + 1]);k--) {
while (!s.empty() && u[k] <= u[s.top()]) s.pop();
r[k] = s.empty() ? j : s.top() - 1;///同上
s.push(k);
}
j = k + 1;
}
for (int j = 1;j <= m;j++) {
int a = r[j] - l[j] + 1;
ans1 = max(ans1, min(a, u[j]) * min(a, u[j]));
ans2 = max(ans2, a * u[j]);
}
}
cout << ans1 << '\n' << ans2 << '\n';
return 0;
}
NC20471 [ZJOI2007]棋盘制作的更多相关文章
- 洛谷 P1169 [ZJOI2007]棋盘制作
2016-05-31 14:56:17 题目链接: 洛谷 P1169 [ZJOI2007]棋盘制作 题目大意: 给定一块矩形,求出满足棋盘式黑白间隔的最大矩形大小和最大正方形大小 解法: 神犇王知昆的 ...
- BZOJ1057 [ZJOI2007]棋盘制作(极大化思想)
1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec Memory Limit: 162 MB Submit: 1848 Solved: 936 [Submit][Sta ...
- bzoj 1057: [ZJOI2007]棋盘制作 单调栈
题目链接 1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 2027 Solved: 1019[Submit] ...
- BZOJ 1057: [ZJOI2007]棋盘制作( dp + 悬线法 )
对于第一问, 简单的dp. f(i, j)表示以(i, j)为左上角的最大正方形, f(i, j) = min( f(i + 1, j), f(i, j + 1), f(i + 1, j + 1)) ...
- 悬线法 || BZOJ 1057: [ZJOI2007]棋盘制作 || Luogu P1169 [ZJOI2007]棋盘制作
题面:P1169 [ZJOI2007]棋盘制作 题解: 基本是悬线法板子,只是建图判断时有一点点不同. 代码: #include<cstdio> #include<cstring&g ...
- P1169 [ZJOI2007]棋盘制作 && 悬线法
P1169 [ZJOI2007]棋盘制作 给出一个 \(N * M\) 的 \(01\) 矩阵, 求最大的正方形和最大的矩形交错子矩阵 \(n , m \leq 2000\) 悬线法 悬线法可以求出给 ...
- [luogu P1169] [ZJOI2007]棋盘制作
[luogu P1169] [ZJOI2007]棋盘制作 题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的 ...
- 1057: [ZJOI2007]棋盘制作
1057: [ZJOI2007]棋盘制作 https://www.lydsy.com/JudgeOnline/problem.php?id=1057 分析: 首先对于(i+j)&1的位置0-& ...
- 【BZOJ 1057】 1057: [ZJOI2007]棋盘制作
1057: [ZJOI2007]棋盘制作 Description 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源 于易经的思想,棋盘是一个8*8大小的 ...
随机推荐
- Map和WeakMap的方法和区别
Map Map是一组键值对的结构,具有极快的查找速度. 一.构造函数不同 let map = new Map(); let weakmap = new WeakMap(); 二.内置函数不同 Map的 ...
- 基于SqlSugar的开发框架循序渐进介绍(6)-- 在基类接口中注入用户身份信息接口
在基于SqlSugar的开发框架中,我们设计了一些系统服务层的基类,在基类中会有很多涉及到相关的数据处理操作的,如果需要跟踪具体是那个用户进行操作的,那么就需要获得当前用户的身份信息,包括在Web A ...
- String 为什么不可变?
转载来源:String为什么不可变 今天来分享一道群友去阿里云面试遇到的 Java 基础面试真题:"String.StringBuffer.StringBuilder 的区别?String ...
- 前端2CSS
内容概要 form表单 网络请求方式 CSS简介 CSS查找标签之基本选择器(重要) CSS查找标签之组合选择器(重要) 属性选择器 分组与嵌套 伪类选择器 内容详情 form表单 "&qu ...
- SQL的语法
SQL的语法 SQL通用语法 SQL语句可以单行或多行书写,以分号(";")结尾. SQL语句可以使用空格或缩进增强可读性. MySQL数据库的SQL语句不区分大小写(建议关键字大 ...
- markdown常用到的语法
一.标题 后加文字,几个#代表几级标题,最高为6 ,标准语法一般在#后跟个空格再写文字. 二.分割线 三个或者三个以上的 - 或者 * 三.图片 格式: ![A](B "C") A ...
- 有关安装pycocotools的办法
1,首先需要安装Visual C++ 2015构建工具,地址https://download.microsoft.com/download/5/f/7/5f7acaeb-8363-451f-9425- ...
- JS:eval
定义和用法: eval() 函数计算 JavaScript 字符串,并把它作为脚本代码来执行.eval()函数并不会创建一个新的作用域,并且它的作用域就是它所在的作用域. 如果参数是一个表达式,eva ...
- 记一次APP渗透登录验证绕过思路
前言: 起初是抓包时候查看返回状态码不一致,所以觉得是否可以通过修改状态码来达到绕过的目的,但是拦截响应包再替换手速不够,技术大哥就去搜了下,找到了一个方法,可以自动替换响应包内容. 在偏下方一点的地 ...
- Linux安装Anaconda3完整教程
Linux安装Anaconda3完整教程 欢迎关注H寻梦人公众号 相关链接 官方安装Anaconda3教程 [手把手教你]如何在Linux系统搭建jupyter notebook CentOS8.2安 ...