输入包含了多个游戏棋盘。每个棋盘包含了 n^2 个点的正方形矩阵 (其中 2 ≤ n ≤ 9),以及一些起连接作用的横向或纵向的线段。棋盘的的 n^2 个点和 m 条连接线段,格式如下:
第 1 行:n,表示矩阵中单行或单列的点的数目 第 2 行:m,表示连接线段的数目 接下来的 m 行,每行是以下两种格式之一: (1) H i j 形式,表示第 i 行的横向线段,连接了第 j 列的点和它右边的第 j + 1 列的点; (2) V i j 形式,表示第 i 列的纵向线段,连接了第 j 行的点和它下方的第 j + 1 行的点。
样例输入数据的第 1 组,对应于上面的图示棋盘。
对于每组测试数据,输出 Problem #1, Problem #2 等标识,并输出棋盘上的各种大小的正方形数目,按正方形由小到大的顺序排列。如果不存在任何大小的正方形,则打印相应的提示消息。将各组测试数据以一行星号间隔,星号上下方各有一个空行。请参见示例的格式。
输入 #1
4 16 H 1 1 H 1 3 H 2 1 H 2 2 H 2 3 H 3 2 H 4 2 H 4 3 V 1 1 V 2 1 V 2 2 V 2 3 V 3 2 V 4 1 V 4 2 V 4 3 2 3 H 1 1 H 2 1 V 2 1
输出 #1
Problem #1 2 square (s) of size 1 1 square (s) of size 2 ********************************** Problem #2 No completed squa
分开存横边和竖边,先查size为1的正方形,从第一个顶点开始查,具体思路可以看代码
#include <stdio.h> #include <string.h> #define maxn 15 // 分开储存 int r[maxn][maxn]; int c[maxn][maxn]; int main() { int n; int kase = 0; while (scanf("%d", &n) != EOF) { memset(r, 0, sizeof(r)); memset(c, 0, sizeof(c)); // 注意格式!!! if (kase > 0) { printf("\n"); printf("**********************************\n\n"); } int len; scanf("%d", &len); for (int k = 0; k < len; k++) { char ch; int i, j; getchar(); scanf("%c", &ch); if (ch == 'H') { scanf("%d%d", &i, &j); // 有边则设置为1 r[i][j] = 1; } else if (ch == 'V') { scanf("%d%d", &i, &j); c[i][j] = 1; } } int flag = 1; int all[n]; memset(all, 0, sizeof(all)); printf("Problem #%d\n\n", ++kase); int num = 0; for (int i = 1; i < n; i++) { for (int j = 1; j <= n * (n - i); j++) { flag = 1; for (int k = 0; k < i; k++) { // 关键代码,判断边是否符合条件. if (r[(j - 1) / n + 1][(j - 1) % n + 1 + k] != 1 || r[(j - 1) / n + 1 + i][(j - 1) % n + 1 + k] != 1|| c[(j - 1) % n + 1][(j - 1) / n + 1 + k] != 1 ||c[(j - 1) % n + 1 + i][(j - 1) / n + 1 + k] != 1) { flag = 0; break; } } if (flag == 1) { all[i]++; } } if (all[i] > 0) { printf("%d square (s) of size %d\n", all[i], i); num++; } } if (num == 0) { printf("No completed squares can be found.\n"); } } return 0; }