[BZOJ4572][Scoi2016]围棋

[BZOJ4572][Scoi2016]围棋

试题描述

近日,谷歌研发的围棋AI—AlphaGo以4:1的比分战胜了曾经的世界冠军李世石,这是人工智能领域的又一里程碑。与传统的搜索式AI不同,AlphaGo使用了最近十分流行的卷积神经网络模型。在卷积神经网络模型中,棋盘上每一块特定大小的区域都被当做一个窗口。例如棋盘的大小为5×6,窗口大小为2×4,那么棋盘中共有12个窗口。此外,模型中预先设定了一些模板,模板的大小与窗口的大小是一样的。下图展现了一个5×6的棋盘和两个2×4的模板。对于一个模板,只要棋盘中有某个窗口与其完全匹配,我们称这个模板是被激活的,否则称这个模板没有被激活。例如图中第一个模板就是被激活的,而第二个模板就是没有被激活的。我们要研究的问题是:对于给定的模板,有多少个棋盘可以激活它。为了简化问题,我们抛开所有围棋的基本规则,只考虑一个n×m的棋盘,每个位置只能是黑子、白子或无子三种情况,换句话说,这样的棋盘共有3n×m种。此外,我们会给出q个2×c的模板。我们希望知道,对于每个模板,有多少种棋盘可以激活它。强调:模板一定是两行的。

输入

输入数据的第一行包含四个正整数n,m,c和q,分别表示棋盘的行数、列数、模板的列数和模板的数量。随后2×q行,每连续两行描述一个模板。其中,每行包含c个字符,字符一定是‘W’,‘B’或‘X’中的一个,表示白子、黑子或无子三种情况的一种。N<=100,M<=12,C<=6,Q<=5

输出

输出应包含q行,每行一个整数,表示符合要求的棋盘数量。由于答案可能很大,你只需要输出答案对1,000,000,007取模后的结果即可。

输入示例

3 1 1 2
B
W
B
B

输出示例

6
5

数据规模及约定

见“输入

题解

看到模板只有两行,就可以考虑用轮廓线 dp 做了。轮廓线的状态就一定分两行,对于上一行我们只需要关心是否完全匹配了模板的第一行就好了。

于是设计出状态 f(i, j, S, p1, p2) 表示现在在表格第 i 行第 j 列的位置,轮廓线状态为 S(如果某一位上二进制是 1 则表示这一位往前数 C 位得到的串与模板第一行完全相同),p1 和 p2 分别表示当前行已经填的串和模板的第一行、第二行的 KMP 上的匹配位置。转移显然。

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cctype>
#include <algorithm>
using namespace std;

int read() {
	int x = 0, f = 1; char c = getchar();
	while(!isdigit(c)){ if(c == '-') f = -1; c = getchar(); }
	while(isdigit(c)){ x = x * 10 + c - '0'; c = getchar(); }
	return x * f;
}

#define maxn 110
#define maxm 15
#define maxc 10
#define maxs 4096
#define MOD 1000000007
#define LL long long

int n, m, C, f[2][maxm][maxs][maxc][maxc];
struct KMP {
	int S[maxc], f[maxc], to[maxc][3];
	void init() {
		memset(f, 0, sizeof(f));
		return ;
	}
	void getstr() {
		char c = getchar();
		while(!isalpha(c)) c = getchar();
		for(int i = 1; i <= C; i++, c = getchar())
			if(c == 'X') S[i] = 0;
			else if(c == 'B') S[i] = 1;
			else S[i] = 2;
		for(int i = C + 1; i < maxc; i++) S[i] = 233;
		return ;
	}
	void getfail() {
		f[1] = f[2] = 1;
		for(int i = 2; i <= C; i++) {
			int j = f[i];
			while(j > 1 && S[j] != S[i]) j = f[j];
			f[i+1] = S[j] == S[i] ? j + 1 : 1;
		}
		for(int i = 1; i <= C + 1; i++)
			for(int c = 0; c < 3; c++) {
				int j = i;
				while(j > 1 && S[j] != c) j = f[j];
				to[i][c] = S[j] == c ? j + 1 : 1;
			}
		return ;
	}
} l1, l2;

void up(int& a, int b) {
	a += b; if(a >= MOD) a -= MOD;
	return ;
}

int main() {
	n = read(); m = read(); C = read(); int q = read();
	while(q--) {
		l1.init(); l1.getstr(); l1.getfail();
		l2.init(); l2.getstr(); l2.getfail();
		
		memset(f, 0, sizeof(f));
		int all = (1 << m) - 1; bool cur = 0;
		f[cur][1][0][1][1] = 1;
		for(int i = 1; i <= n; i++, cur ^= 1) {
			memset(f[cur^1], 0, sizeof(f[cur^1]));
			int tmp = 0;
			for(int S = 0; S <= all; S++) up(tmp, f[cur][1][S][1][1]);
			for(int j = 1; j <= m; j++)
				for(int S = 0; S <= all; S++)
					for(int p1 = 1; p1 <= C + 1; p1++)
						for(int p2 = 1; p2 <= C + 1; p2++) {
							int& ans = f[cur][j][S][p1][p2];
							if(!ans) continue;
							for(int c = 0; c < 3; c++) {
								int t1 = l1.to[p1][c], t2 = l2.to[p2][c];
								if((S & 1) && t2 == C + 1) continue;
								int tS = S >> 1 | ((t1 == C + 1) << m - 1);
								if(j < m) up(f[cur][j+1][tS][t1][t2], ans);
								else up(f[cur^1][1][tS][1][1], ans);
							}
						}
		}
		int ans = 0; LL sum = 1;
		for(int S = 0; S <= all; S++) up(ans, f[cur][1][S][1][1]);
		for(int i = 1; i <= n * m; i++) (sum *= 3) %= MOD;
		printf("%d\n", (int)((sum - ans + MOD) % MOD));
	}
	
	return 0;
}

 

posted @ 2017-06-13 18:26  xjr01  阅读(377)  评论(0编辑  收藏  举报