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大小的 ...
随机推荐
- 基于surging网络组件多协议适配的平台化发展
前言 Surging 发展已经有快6年的时间,经过这些年的发展,功能框架也趋于成熟,但是针对于商业化需求还需要不断的打磨,前段时间客户找到我想升级成平台化,针对他的需求我 ...
- POJ1821 Fence 题解报告
传送门 1 题目描述 A team of $k (1 <= K <= 100) $workers should paint a fence which contains \(N (1 &l ...
- nvm安装与使用及乱码问题
前端开发工作中经常负责多个项目(新项目.多年的老项目及团队合作项目),经常会遇到npm install安装依赖包或者启动本地服务时依赖报错的情况,大多数是因为NodeJS和npm与依赖之间版本的问题, ...
- [WUSTCTF2020]颜值成绩查询-1
分享下自己在完成[WUSTCTF2020]颜值成绩查询-1关卡的手工过程和自动化脚本. 1.通过payload:1,payload:1 ,payload:1 or 1=1--+,进行判断是否存在注入, ...
- 【翻译】驯服野兽:Scylla 如何利用控制理论来控制压实
教程翻译自Seastar官方文档:https://www.scylladb.com/2018/06/12/scylla-leverages-control-theory/ 转载请注明出处:https: ...
- redis入门,linux安装
1.下载 https://redis.io/download 2.上传到linux服务器tools文件夹下 3.解压到安装目录 tar -zxf /app/redis/redis-5.0.4.tar. ...
- 接口偶尔超时,竟又是JVM停顿的锅!
原创:扣钉日记(微信公众号ID:codelogs),欢迎分享,转载请保留出处. 简介 继上次我们JVM停顿十几秒的问题解决后,我们系统终于稳定了,再也不会无故重启了! 这是之前的文章:耗时几个月,终于 ...
- 1.1 操作系统的第一个功能——虚拟化硬件资源 -《zobolの操作系统学习札记》
1.1 操作系统的第一个功能--虚拟化硬件资源 目录 1.1 操作系统的第一个功能--虚拟化硬件资源 问1:操作系统一般处于计算机系统的哪一个位置? 问2:管理硬件资源为什么要单独交给操作系统? 问3 ...
- JS:条件语句2
1.for循环:循环代码块一定的次数 例: for(var i = 0;i<5;i++){ console.log(i); } // 0 1 2 3 4 遍历对象: var arr=[" ...
- 写个js获取2019博客之星投票活动的名次与投票数
获取投票数 // app.jsvar request = require('request');var cheerio = require('cheerio');request('http://m23 ...