[洛谷P1169] [ZJOI2007] 棋盘制作 解题报告(悬线法+最大正方形)
题目描述
国际象棋是世界上最古老的博弈游戏之一,和中国的围棋、象棋以及日本的将棋同享盛名。据说国际象棋起源于易经的思想,棋盘是一个 8×8 大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳。
而我们的主人公小Q,正是国际象棋的狂热爱好者。作为一个顶尖高手,他已不满足于普通的棋盘与规则,于是他跟他的好朋友小W决定将棋盘扩大以适应他们的新规则。
小Q找到了一张由 N×M 个正方形的格子组成的矩形纸片,每个格子被涂有黑白两种颜色之一。小Q想在这种纸中裁减一部分作为新棋盘,当然,他希望这个棋盘尽可能的大。
不过小Q还没有决定是找一个正方形的棋盘还是一个矩形的棋盘(当然,不管哪种,棋盘必须都黑白相间,即相邻的格子不同色),所以他希望可以找到最大的正方形棋盘面积和最大的矩形棋盘面积,从而决定哪个更好一些。
于是小Q找到了即将参加全国信息学竞赛的你,你能帮助他么?
输入输出格式
输入格式:
包含两个整数 N 和 M ,分别表示矩形纸片的长和宽。接下来的 NN 行包含一个 N ×M 的 01 矩阵,表示这张矩形纸片的颜色( 0 表示白色, 1 表示黑色)。
输出格式:
包含两行,每行包含一个整数。第一行为可以找到的最大正方形棋盘的面积,第二行为可以找到的最大矩形棋盘的面积(注意正方形和矩形是可以相交或者包含的)。
输入输出样例
3 3
1 0 1
0 1 0
1 0 0
4
6 对于这道题目,首先需要一个脑洞很大的转换——“如果横坐标加纵坐标是偶数那么取反,然后转化为最大子矩形”
下面我拿样例做个例子
对于样例矩阵 1 0 1
0 1 0
1 0 0
经过转化后就变成了
0 0 0
0 0 0
0 0 1
最大子矩形面积为6,最大正方形面积为4
为什么这样可以呢?我的理解是,如果一个矩形可以作为棋盘,那么它的黑白两色肯定是相间的。既然是相间的,每隔一位颜色取反岂不就是同一种颜色了吗?当然,黑白二色要各判断一次最大子矩形和最大子正方形。 下面我们考虑怎么得到最大子正方形,
f[i][j]表示从第i行第j列向第一行第一列逐渐扩展的最大子正方形的边长。在a[i][j]为我们统计的数(1或0)的情况下,f[i][j]=min(f[i-1][j-1],min(f[i-1][j],f[i][j-1])+1
第一次看到的人估计很难理解,为什么是min呢?确定没有打错吗?然而,没错,当时我也花了很久才能理解
下面写一个例子,我们计算1的最大正方形
1 1 1 0 1 1
0 1 1 1 1 0
1 1 1 1 1 0
0 0 1 1 1 1
1 1 1 1 0 1
f[1][1]=1 f[1][2]=1 f[2][1]=0 f[2][2]=min(f[1][2],f[1][1],f[2][1])+1=1
f[1][3]=1 f[1][4]=0 f[1][5]=1 f[1][6]=1
f[2][3]=min(f[1][3],f[1][2],f[2][2])+1=2
.......
下面的读者可以手推。
下面我讲讲自己的理解。因为要形成一个正方形,f[i-1][j-1],f[i-1][j],f[i][j-1]必须要同时满足“条件”,即子正方形内必须完全是1,有一个是0都不行,这就是为什么取min了 下面我们考虑怎么得到最大子矩形(下面只讨论统计1的,统计0的同理)
1.预处理出
l[i][j]以(i,j)向左一直为1的话最大扩展到第几列
r[i][j]以(i,j)向右一直为1的话最大扩展到第几列
不理解的话看看程序就懂了
2.
L[i][j]以(i,j)为下端点的悬线向左一直为1的话最大扩展到第几列
R[i][j]以(i,j)为下端点的悬线向右一直为1的话最大扩展到第几列
h[i][j],表示(i,j)为下端点的最大悬线长(就是(i,j)向上始终为1最大能有多长)
对于(i,j)表示的最大矩形就是(R[i][j]-L[i][j]-1)*h[i][j] 还有一点小细节见注释
#include<bits/stdc++.h>
#define mem(aa,bb) memset(aa,bb,sizeof(aa))
#define ri register int
using namespace std; const int maxn=+;
int n,m,ans1,ans2;
int a[maxn][maxn],f[maxn][maxn],l[maxn][maxn],r[maxn][maxn],L[maxn][maxn],R[maxn][maxn],h[maxn][maxn];
void work()
{
mem(f,);mem(l,);mem(r,);mem(L,);mem(R,);mem(h,);
for (ri i=;i<=n;i++)//预处理
{
int tmp=;
for (ri j=;j<=m;j++)
if (a[i][j]) l[i][j]=tmp;
else tmp=j,L[i][j]=;
tmp=m+;
for (ri j=m;j>=;j--)
if (a[i][j]) r[i][j]=tmp;
else tmp=j,R[i][j]=m+;
}
for (int i=;i<=m;i++) R[][i]=m+;//小细节
for (ri i=;i<=n;i++)
for (ri j=;j<=m;j++)
if (a[i][j]){
f[i][j]=min(f[i-][j-],min(f[i-][j],f[i][j-]))+;
h[i][j]=h[i-][j]+;//采取一边计算一边处理
L[i][j]=max(L[i-][j],l[i][j]);
R[i][j]=min(R[i-][j],r[i][j]);
ans1=max(ans1,f[i][j]*f[i][j]);
ans2=max(ans2,h[i][j]*(R[i][j]-L[i][j]-));
}
}
int main()
{
scanf("%d%d",&n,&m);
for (ri i=;i<=n;i++)
for (ri j=;j<=m;j++)
{
scanf("%d",&a[i][j]);
if (!((i+j)&)) a[i][j]=-a[i][j];
}
work();
for (ri i=;i<=n;i++)
for (ri j=;j<=m;j++)
a[i][j]=-a[i][j];
work();
printf("%d\n%d",ans1,ans2);
return ;
}
[洛谷P1169] [ZJOI2007] 棋盘制作 解题报告(悬线法+最大正方形)的更多相关文章
- 洛谷 P1169 [ZJOI2007]棋盘制作
2016-05-31 14:56:17 题目链接: 洛谷 P1169 [ZJOI2007]棋盘制作 题目大意: 给定一块矩形,求出满足棋盘式黑白间隔的最大矩形大小和最大正方形大小 解法: 神犇王知昆的 ...
- 洛谷P1169 [ZJOI2007]棋盘制作 悬线法 动态规划
P1169 [ZJOI2007]棋盘制作 (逼着自己做DP 题意: 给定一个包含0,1的矩阵,求出一个面积最大的正方形矩阵和长方形矩阵,要求矩阵中相邻两个的值不同. 思路: 悬线法. 用途: 解决给定 ...
- 【题解】洛谷P1169 [ZJOI2007] 棋盘制作(坐标DP+悬线法)
次元传送门:洛谷P1169 思路 浙江省选果然不一般 用到一个从来没有听过的算法 悬线法: 所谓悬线法 就是用一条线(长度任意)在矩阵中判断这条线能到达的最左边和最右边及这条线的长度 即可得到这个矩阵 ...
- 洛谷 P1169 [ZJOI2007]棋盘制作 (悬线法)
和玉蟾宫很像,条件改成不相等就行了. 悬线法题目 洛谷 P1169 p4147 p2701 p1387 #include<cstdio> #include<algorithm& ...
- 洛谷P1169[ZJOI2007]棋盘制作
题目 一道悬线法的裸题,悬线法主要是可以处理最大子矩阵的问题. 而这道题就是比较经典的可以用悬线法来处理的题. 而悬线法其实就是把矩阵中对应的每个位置上的元素分别向左向上向右,寻找到不能到达的地方,然 ...
- BZOJ1057或洛谷1169 [ZJOI2007]棋盘制作
BZOJ原题链接 洛谷原题链接 设\(L[i][j],R[i][j],H[i][j]\)表示点\((i,j)\)向左.右.上尽量拓展的左端点.右端点.上端点的坐标. \(L,R\)直接初始化好,\(H ...
- 洛谷 P1129 [ZJOI2007]矩阵游戏 解题报告
P1129 [ZJOI2007]矩阵游戏 题目描述 小\(Q\)是一个非常聪明的孩子,除了国际象棋,他还很喜欢玩一个电脑益智游戏――矩阵游戏.矩阵游戏在一个\(N*N\)黑白方阵进行(如同国际象棋一般 ...
- 洛谷 P1110 [ZJOI2007]报表统计 解题报告
P1110 [ZJOI2007]报表统计 题目描述 \(Q\)的妈妈是一个出纳,经常需要做一些统计报表的工作.今天是妈妈的生日,小\(Q\)希望可以帮妈妈分担一些工作,作为她的生日礼物之一. 经过仔细 ...
- 洛谷1169 [ZJOI2007] 棋盘制作
题目链接 题意概述:给出由0 1构成的矩阵,求没有0 1 相邻的最大子矩阵的最大子正方形. 解题思路:设f[i][j]表示i j向上能到哪,l[i][j] r[i][j]表示向左/右,转移时分开计算矩 ...
随机推荐
- [Oracle] Merge语句
Merge的语法例如以下: MERGE [hint] INTO [schema .] table [t_alias] USING [schema .] { table | view | subquer ...
- POJ 2154
这题的时间卡的.... 必须用欧拉来优化,而且要加素数表.最重要是,因为最后结果要/n,而数据很大,所以,必须在之前就先/n了,否则会爆数据. #include <iostream> #i ...
- oracle实现自增id
--oracle实现自增id --创建一张T_StudentInfo表 create table T_StudentInfo ( "id" integer not null pri ...
- L2CAP数据发送和接收
ACL 链路在 Bluetooth 中非常重要,一些重要的应用如 A2DP, 基于 RFCOMM 的应用.BNEP等都要建立 ACL 链路,发送/接收ACL 包.跟大家一起来分析 ACL 包发送/接收 ...
- nyoj--61--传纸条(一)(动态规划)
传纸条(一) 时间限制:2000 ms | 内存限制:65535 KB 难度:5 描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列 ...
- MHA+ProxySQL 读写分离高可用
文档结构如下: 1.ProxySQL说明 ProxySQL是mysql的一款中间件的产品,是灵活的mysql代理层,可以实现读写分离,支持query路由器的功能,支持动态指定sql进行缓存,支持动态加 ...
- BZOJ 3339 线段树
思路: 考虑离线处理 显然 l固定时 r越大 ans越大 那我们不妨按照l从小到大排序 l->l+1的时候 l到next[l]这段区间都跟a[l]取min就好了 搞颗线段树维护一下 //By S ...
- Windows下本机简易监控系统搭建(Telegraf+Influxdb+Grafana)--转
原文地址:http://www.cnblogs.com/liugh/p/6683488.html 一.文件准备 1.1 文件名称 telegraf-1.2.1_windows_amd64.zip in ...
- 【英雄会】微软题目:几个bing
今天是元旦,开篇先祝福大家在新的一年心想事成,工作顺利,开心生活每一天 . 看到[英雄会]上出现了微软出的题目:几个bing,题目内容如下: 本届大赛由微软必应词典冠名,必应词典(Bing Dicti ...
- Glidar测试安装
在上一篇随笔中,我们完成了对Glidar 仿真器的概念层面的认识.接下来,我们将着手对该该仿真器进行安装测试. 1 依赖库的安装 安装环境为Windows 7 64位+Ubuntu14.04 LTS的 ...