Description

国际象棋是世界上最古老的博弈游戏之一,和中国的围棋、象棋以及日本的将棋同享盛名。据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳。而我们的主人公小Q,正是国际象棋的狂热爱好者。作为一个顶尖高手,他已不满足于普通的棋盘与规则,于是他跟他的好朋友小W决定将棋盘扩大以适应他们的新规则。小Q找到了一张由N*M个正方形的格子组成的矩形纸片,每个格子被涂有黑白两种颜色之一。小Q想在这种纸中裁减一部分作为新棋盘,当然,他希望这个棋盘尽可能的大。不过小Q还没有决定是找一个正方形的棋盘还是一个矩形的棋盘(当然,不管哪种,棋盘必须都黑白相间,即相邻的格子不同色),所以他希望可以找到最大的正方形棋盘面积和最大的矩形棋盘面积,从而决定哪个更好一些。于是小Q找到了即将参加全国信息学竞赛的你,你能帮助他么?

Input

第一行包含两个整数N和M,分别表示矩形纸片的长和宽。接下来的N行包含一个N * M的01矩阵,表示这张矩形纸片的颜色(0表示白色,1表示黑色)。

Output

包含两行,每行包含一个整数。第一行为可以找到的最大正方形棋盘的面积,第二行为可以找到的最大矩形棋盘的面积(注意正方形和矩形是可以相交或者包含的)。

Sample Input

3 3
1 0 1
0 1 0
1 0 0

Sample Output

4
6

HINT

对于100%的数据,N, M ≤ 2000

 
首先看肯定要转换成最大0/1子矩阵,但是怎么转换呢???
这个做法很赞。将矩阵进行黑白染色后,依题意棋盘需要黑白相间,就是相邻的黑白格子颜色互不相同。假设我们将黑色(白色)格子的值取反后,合法的状态即为黑白格子颜色相同了,就是求一个最大0/1子矩阵。。。
那么最大0/1子矩阵怎么在O(N*M)的时间内求出,我们可以dp。up[i][j]表示从(i,j)最高可以伸长几个格子,le[i][j],ri[i][j]表示up[i][j]这根悬线可以最左最右移动到哪里,最大的矩形的面基ans1=max(up[i][j]*(ri[i][j]-le[i][j]+1),ans1),最大方阵面积ans2=max(min(up[i][j],(ri[i][j]-le[i]))2,ans2)。对0和1各做一遍即可。
转移很好写:
 for (int i = ;i <= n;++i)
{
int lo = ,ro = m+;
for (int j = ;j <= m;++j)
{
if (s[i][j] == sign) up[i][j] = le[i][j] = ,lo = j;
else up[i][j] = i==?:up[i-][j]+,le[i][j] = i==?lo+:max(le[i-][j],lo+);
}
for (int j = m;j;--j)
{
if (s[i][j] == sign) ri[i][j] = m+,ro = j;
else ri[i][j] = i==?ro-:min(ri[i-][j],ro-);
int a = up[i][j],b = ri[i][j]-le[i][j]+,p = min(a,b);
ans1 = max(ans1,p*p); ans2 = max(ans2,a*b);
}
}

总代码:

 #include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std; #define maxn 2010
int s[maxn][maxn],le[maxn][maxn],ri[maxn][maxn];
int up[maxn][maxn],n,m,ans1,ans2; inline void work()
{
for (int i = ;i <= n;++i)
{
for (int j = ;j <= m;++j)
{
if (i == || s[i][j] == s[i-][j]) up[i][j] = ;
else up[i][j] = up[i-][j]+;
}
}
} inline void deal()
{
for (int i = ;i <= n;++i)
for (int j = ;j <= m;++j)
if ((i + j)&) s[i][j] ^= ;
} inline void work(int sign)
{
for (int i = ;i <= n;++i)
{
int lo = ,ro = m+;
for (int j = ;j <= m;++j)
{
if (s[i][j] == sign) up[i][j] = le[i][j] = ,lo = j;
else up[i][j] = i==?:up[i-][j]+,le[i][j] = i==?lo+:max(le[i-][j],lo+);
}
for (int j = m;j;--j)
{
if (s[i][j] == sign) ri[i][j] = m+,ro = j;
else ri[i][j] = i==?ro-:min(ri[i-][j],ro-);
int a = up[i][j],b = ri[i][j]-le[i][j]+,p = min(a,b);
ans1 = max(ans1,p*p); ans2 = max(ans2,a*b);
}
}
} int main()
{
freopen("1057.in","r",stdin);
freopen("1057.out","w",stdout);
scanf("%d %d",&n,&m);
for (int i = ;i <= n;++i)
for (int j = ;j <= m;++j) scanf("%d",s[i]+j);
deal(); work(); work();
printf("%d\n%d",ans1,ans2);
fclose(stdin); fclose(stdout);
return ;
}

BZOJ 1057 棋盘制作的更多相关文章

  1. BZOJ 1057 棋盘制作(最大01相间子矩阵)

    求最大01相间子矩阵可以转换为求最大全0子矩阵.只需把棋盘(x+y)为奇数的取反,而该问题可以用经典的悬线法O(n^2)的求解. 悬线法呢. 首先定义b[i][j],为a[i][j]向上的最大连续0的 ...

  2. 【以前的空间】bzoj [ZJOI2007]棋盘制作

    具体可以去跪<浅谈用极大化思想解决最大子矩形问题>(p.s. 蒟蒻跪了还是很晕,不过想到之前usaco好像是最后一章的一道题……看了下代码顿然醒悟) 也就是如果用o(nm)的方法维护一个极 ...

  3. bzoj 1057: [ZJOI2007]棋盘制作 单调栈

    题目链接 1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 2027  Solved: 1019[Submit] ...

  4. 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)) ...

  5. 【BZOJ 1057】 1057: [ZJOI2007]棋盘制作

    1057: [ZJOI2007]棋盘制作 Description 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源 于易经的思想,棋盘是一个8*8大小的 ...

  6. BZOJ 1057: [ZJOI2007]棋盘制作 悬线法求最大子矩阵+dp

    1057: [ZJOI2007]棋盘制作 Description 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑 ...

  7. 悬线法 || BZOJ 1057: [ZJOI2007]棋盘制作 || Luogu P1169 [ZJOI2007]棋盘制作

    题面:P1169 [ZJOI2007]棋盘制作 题解: 基本是悬线法板子,只是建图判断时有一点点不同. 代码: #include<cstdio> #include<cstring&g ...

  8. BZOJ 1057:[ZJOI2007]棋盘制作(最大01子矩阵+奇偶性)

    [ZJOI2007]棋盘制作                                          时间限制: 20 Sec 内存限制: 162 MB[题目描述]国际象棋是世界上最古老的博 ...

  9. 棋盘制作 BZOJ 1057

    棋盘制作 [问题描述] 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴 ...

随机推荐

  1. 基于JAX-WS的Web Service服务端/客户端 ;JAX-WS + Spring 开发webservice

    一.基于JAX-WS的Web Service服务端/客户端 下面描述的是在main函数中使用JAX-WS的Web Service的方法,不是在web工程里访问,在web工程里访问,参加第二节. JAX ...

  2. Linux下安装 TestLink常见问题解决方法

    Read/write permissions For security reason we suggest that directories tagged with [S] on following ...

  3. [MODx] Solve cannot upload large file

    If you also run into this problem, dont' worry, here is the solution for you. First: In Modx, go &qu ...

  4. 读取一个文件,将其Base64编码,每76个字符加一个换行(转)

    echo chunk_split(base64_encode(file_get_contents('base64.txt'))); 例子 1 本例分隔每个字符,并添加 ".": & ...

  5. 如何设计一个简单的C++ ORM

    2016/11/15 "没有好的接口,用C++读写数据库和写图形界面一样痛苦" 阅读这篇文章前,你最好知道什么是 Object Relation Mapping (ORM) 阅读这 ...

  6. android开发之AlertDialog点击按钮之后不消失 分类: android 学习笔记 2015-07-15 18:07 89人阅读 评论(0) 收藏

    最近有这样一个需求,我需要用户在一个弹出框里输入密码来验证,验证成功当然好说,但是如果验证失败则需要把alertdialog的标题改为"密码错误,请重新输入",并且这个alertd ...

  7. Configuring Robolectric

    There are numerous ways to customize how Robolectric behaves at runtime. Config Annotation The prima ...

  8. PHP错误类型及屏蔽方法

    1. 注意(Notices)这些都是比较小而且不严重的错误,比如去访问一个未被定义的变量.通常,这类的错误是不提示给用户的,但有时这些错误会影响到运行的结果. 2. 警告(Warnings)这就是稍微 ...

  9. 自定义图文混排视图MyImageTextView

    http://blog.csdn.net/xujunfeng000/article/details/36399339?utm_source=tuicool&utm_medium=referra ...

  10. HashMap深度解析(一)

    本文来自:高爽|Coder,原文地址:http://blog.csdn.net/ghsau/article/details/16843543 HashMap可以说是Java中最常用的集合类框架之一,是 ...