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. Access中出现改变字段“自己主动编号”类型,不能再改回来!(已解决)

    Access中出现改变字段"自己主动编号"类型,不能再改回来! (已解决) 一次把access中的自增字段改成了数值,再改回自增时,提示:在表中输入了数据之后,则不能将不论什么字段 ...

  2. 对vpn的认识

    网上关于vpn的资料非常多,看后眼花缭乱,仍然感觉一片混沌.网上,网下參考一些资料后.试着理清一些概念问题,由于,概念理清了,找到门了,才不至于左右徘徊,一片混沌. 首先vpn,这个我们都知道叫虚拟专 ...

  3. POJ 3094 Quicksum(简单的问题)

    [简要题意]:题意是非常easy. 看样能理解 [分析]:略. 读取字符串. // 200K 0Ms #include<iostream> using namespace std; int ...

  4. PHP手机获取6为不反复验证码

    //存数字数组 $code = array(); while(count($code) < 6){ //产生随机数1-9 $code[] = rand(1,9); //去除数组中的反复元素   ...

  5. iOS-UITableCell详情

    iOS-UITableCell详情 表示UITableViewCell风格的常量有: UITableViewCellStyleDefault UITableViewCellStyleSubtitle ...

  6. iOS NavigaitonController详解(code版)

    参考文章:http://blog.csdn.net/totogo2010/article/details/7681879,参考了这篇文章,写的超级好,自己他的基础上加上了自己的理解. 下面的图显示了导 ...

  7. Java——(三)Collection之Set集合、HashSet类

    ------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 一.Set集合 Set集合不允许包含相同的元素,如果试图把两个相同的元素加入同一个Set集合中, ...

  8. HttpContext.Current

    HttpContext. Response 直接这样写会报错 是因为 httpcontext没有提供response 这个静态的方法. 通过这样写就可以 ASP.NET还为它提供了一个静态属性Http ...

  9. 国人编写的开源 .net Ioc 框架——My.Ioc 简介

    My.Ioc 是作者开发的一款开源 IoC/DI 框架,下载地址在此处.它具有下面一些特点: 高效 在实现手段上,My.Ioc 通过使用泛型.缓存.动态生成代码.延迟注册.尽量使用抽象类而非接口等方式 ...

  10. PHP环境搭建所遇到的问题

    下午学校的机房搭建PHP组合包appserv开发环境的时候是没有任何问题的,但是到了自己的电脑上以后下砸的32位appserve一直无法正常由浏览器的localhos或者127.0.0.1 进入其ap ...