【问题描述 】
国际象棋是世界上最古老的博弈游戏之一,和中国的围棋、象棋以及日本的将棋同享盛名。据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳。
而我们的主人公小Q,正是国际象棋的狂热爱好者。作为一个顶尖高手,他已不满足于普通的棋盘与规则,于是他跟他的好朋友小W决定将棋盘扩大以适应他们的新规则。
小Q找到了一张由N*M个正方形的格子组成的矩形纸片,每个格子被涂有黑白两种颜色之一。小Q想在这种纸中裁减一部分作为新棋盘,当然,他希望这个棋盘尽可能的大。
不过小Q还没有决定是找一个正方形的棋盘还是一个矩形的棋盘(当然,不管哪种,棋盘必须都黑白相间,即相邻的格子不同色),所以他希望可以找到最大的正方形棋盘面积和最大的矩形棋盘面积,从而决定哪个更好一些。
于是小Q找到了即将参加全国信息学竞赛的你,你能帮助他么?
【输入格式 Input Format 】
第一行包含两个整数N和M,分别表示矩形纸片的长和宽。接下来的N行包含一个N * M的01矩阵,表示这张矩形纸片的颜色(0表示白色,1表示黑色)。
【输出格式 Output Format】
包含两行,每行包含一个整数。第一行为可以找到的最大正方形棋盘的面积,第二行为可以找到的最大矩形棋盘的面积(注意正方形和矩形是可以相交或者包含的)。
【样例输入 Sample Input】
3 3
1 0 1
0 1 0
1 0 0
【样例输出 Sample Output】
4
6
注释 Hint
对于20%的数据,N, M ≤ 80
对于40%的数据,N, M ≤ 400
对于100%的数据,N, M ≤ 2000
算法分析
和盖房子那道题目差不多吧.
正方形是一样的.只不过是重新开始的判断条件不同
长方形记录长和宽就好了.
【代码】
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 | #include <stdio.h> #include <string.h> #define MAXN 2001 int a[MAXN][MAXN], f[MAXN][MAXN], c[MAXN][MAXN], r[MAXN][MAXN]; int min(int t1, int t2) { return t1<t2?t1:t2; } int main() { int n, m, i, j, k, maxf = 0, maxc = 0, maxr = 0, s1, s2; scanf("%d%d", &n, &m); memset(f, 0, sizeof(f)); memset(c, 0, sizeof(c)); for (i = 1; i <= n; i++) { for (j = 1; j <= m; j++) scanf("%d", &a[i][j]); } for (i = 1; i<= n; i++) f[i][1] = 1; for (i = 1; i <= m; i++) f[1][i] = 1; for (i = 2; i <= n; i++) //求正方形的连长 { for (j = 2; j<= m; j++) { if (a[i - 1][j] != a[i][j] && a[i][j - 1] != a[i][j]) { f[i][j] = min(f[i - 1][j - 1], min(f[i - 1][j], f[i][j - 1])) + 1; } else { f[i][j] = 1; } if (f[i][j] > maxf) maxf = f[i][j]; } } c[1][1] = 1; r[1][1] = 1; //长宽初始化 for (i = 2; i <= m; i++) { if (a[1][i] != a[1][i - 1]) { c[1][i] = c[1][i - 1] + 1; } else c[1][i] = 1; r[1][i] = 1; } for (i = 2; i <= n; i++) { if (a[i][1] != a[i - 1][1]) { r[i][1] = r[i - 1][1] + 1; } else r[i][1] = 1; c[i][1] = 1; } for (i = 2; i <= n; i++) { for (j = 2; j<= m; j++) { if (a[i - 1][j] != a[i][j] && a[i][j - 1] != a[i][j]) { s1 = min(c[i][j - 1] + 1, c[i - 1][j]) * (r[i - 1][j] + 1); s2 = min(r[i - 1][j] + 1, r[i][j - 1]) * (c[i][j - 1] + 1); if (s1 > s2) { r[i][j] = r[i - 1][j] + 1; c[i][j] = min(c[i][j - 1] + 1, c[i - 1][j]); } else { c[i][j] = c[i][j - 1] + 1; r[i][j] = min(r[i - 1][j] + 1, r[i][j - 1]); } } else { if (a[i - 1][j] != a[i][j]) { r[i][j] = r[i - 1][j] + 1; c[i][j] = 1; } else if (a[i][j - 1] != a[i][j] ) { r[i][j] = 1; c[i][j] = c[i][j - 1] + 1; } else { r[i][j] = 1; c[i][j] = 1; } } if (r[i][j] * c[i][j] > maxr * maxc) { maxr = r[i][j]; maxc = c[i][j]; } } } printf("%d\n%d\n", maxf * maxf, maxr * maxc); return 0; } |