求把 N×MN×M 的棋盘分割成若干个 1×21×2 的长方形,有多少种方案。
例如当 N=2,M=4N=2,M=4 时,共有 55 种方案。当 N=2,M=3N=2,M=3 时,共有 33 种方案。
如下图所示:
输入格式
输入包含多组测试用例。
每组测试用例占一行,包含两个整数 NN 和 MM。
当输入用例 N=0,M=0N=0,M=0 时,表示输入终止,且该用例无需处理。
输出格式
每个测试用例输出一个结果,每个结果占一行。
数据范围
1≤N,M≤111≤N,M≤11
输入样例:
1 2 1 3 1 4 2 2 2 3 2 4 2 11 4 11 0 0输出样例:
1 0 1 2 3 5 144 51205
- #include
- using namespace std;
- const int N = 12, M = 1 << N;
- int st[M];
- long long f[N][M];
-
-
- int main(){
- int n, m;
- while (cin >> n >> m && (n || m)){
-
- for (int i = 0; i < 1 << n; i ++){
- int cnt = 0;
- st[i] = true;
- for (int j = 0; j < n; j ++)
- if (i >> j & 1){
- if (cnt & 1) st[i] = false; // cnt 为当前已经存在多少个连续的0
- cnt = 0;
- }
- else cnt ++;
- if (cnt & 1) st[i] = false; // 扫完后要判断一下最后一段有多少个连续的0
- }
-
- memset(f, 0, sizeof f);
- f[0][0] = 1;
- for (int i = 1; i <= m; i ++)
- for (int j = 0; j < 1 << n; j ++)
- for (int k = 0; k < 1 << n; k ++)
- if ((j & k) == 0 && (st[j | k]))
- // j & k == 0 表示 i 列和 i - 1列同一行不同时捅出来
- // st[j | k] == 1 表示 在 i 列状态 j, i - 1 列状态 k 的情况下是合法的.
- f[i][j] += f[i - 1][k];
- cout << f[m][0] << endl;
- }
- return 0;
- }
-