d(i)表示从i开始的后缀即S[i, L-1]的分解方法数,字符串为S[0, L-1]

则有d(i) = sum{ d(i+len(x)) | 单词x是S[i, L-1]的前缀 }

递推边界为d(L) = 1,代表空串。

将前n个单词构造一颗Tire树,在树中查找后缀的过程中遇到一个单词节点就代表找到一个状态转移中的x

 #include <cstdio>
#include <cstring> const int maxnode = + ;
const int maxl = + ;
const int sigma_size = ;
const int MOD = ; char s[maxnode];
int l;
int d[maxnode];
char s1[maxl]; int sz;
int ch[maxnode][sigma_size];
int val[maxnode]; inline int idx(char c) { return c - 'a'; } void Init() { sz = ; memset(ch[], , sizeof(ch[])); } void insert(char* s, int v)
{
int n = strlen(s), u = ;
for(int i = ; i < n; i++)
{
int c = idx(s[i]);
if(!ch[u][c])
{
memset(ch[sz], , sizeof(ch[sz]));
val[sz] = ;
ch[u][c] = sz++;
}
u = ch[u][c];
}
val[u] = v;
} void solve()
{
for(int i = l-; i >= ; i--)
{
int u = ;
for(int j = i; j < l; j++)
{
int c = idx(s[j]);
if(!ch[u][c]) break;
u = ch[u][c];
if(val[u]) d[i] = (d[i] + d[j + ]) % MOD;
}
}
} int n;
int main()
{
//freopen("in.txt", "r", stdin); int kase = ;
while(scanf("%s", s) == )
{
Init();
l = strlen(s);
scanf("%d", &n);
while(n--) { scanf("%s", s1); insert(s1, ); }//val[i] = 1表示这是一个单词节点
memset(d, , sizeof(d));
d[l] = ;
solve();
printf("Case %d: %d\n", ++kase, d[] % MOD);
} return ;
}

代码君

05-02 03:45