Codeforces 667C DP

题意:给你一个字符串,这个字符串的构造方法如下:先选择一个长度大于4的前缀,然后每次向字符串尾部添加一个长度为2或者长度为3的后缀,不能添加连续的相同的后缀,问可能的后缀有哪些?并按字典序输出去。

思路:第一眼感觉要记忆化,设dp[i]表示把前i个字符作为前缀是否有合法方案,那么只有当前长度为2的串和后面的串重复了,长度为3的串和后面的串重复了,并且dp[i + 5]不合法,dp[i]才不合法。为什么呢?考虑这样一个样例: abcdezzzzzzzz, 合法的后缀有zz 和zzz, 我们可以让zz 和zzz交替出现。

代码:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 10010;
int dp[maxn];
set<string> st;
char s[maxn];
int n;
bool match(int s1, int s2, int cnt) {
	for (int i = 0; i < cnt; i++) {
		if(s[s1 + i] != s[s2 + i]) return 0;
	}
	return 1;
}
int solve(int i) {
	if(i > n) return 0;
	if(dp[i] != -1) {
		return dp[i];
	}
	if(i == n) {
		dp[n] = 1;
		return 1;
	}
	dp[i] = 0;
	if(i + 2 <= n) {
		if(solve(i + 2)) {
			if((i <= n - 3 && match(i + 1, i + 2 + 1, 2)) && !solve(i + 5)) {
					dp[i] |= 0;
			} else {
				dp[i] |= 1;
				string tmp = "";
				tmp += s[i + 1];
				tmp += s[i + 2];
				st.insert(tmp);
			}
		}
	}
	if(i + 3 <= n) {
		if(solve(i + 3)) {
			if((i <= n - 5 && match(i + 1, i + 3 + 1, 3)) && !solve(i + 5)) {
				dp[i] |= 0;
			} else {
				dp[i] |= 1;
				string tmp = "";
				tmp += s[i + 1];
				tmp += s[i + 2];
				tmp += s[i + 3];
				st.insert(tmp);
			}
		}
	}
	return dp[i];
}
int main() {
	scanf("%s",s + 1);
	n = strlen(s + 1);
	if(n <= 5) {
		printf("0\n");
		return 0;
	}
	memset(dp, -1, sizeof(dp));
	for (int i = 5; i <= n; i++)
		solve(i);
	set<string>::iterator it;
	printf("%d\n", st.size());
	for (it = st.begin(); it != st.end(); it++)
		cout << (*it) << endl;
} 

  

posted @ 2019-04-23 08:13  维和战艇机  阅读(227)  评论(0编辑  收藏  举报