HDU2825 Wireless Password(AC自动机+状压DP)

题目问长度n至少包含k个咒语的字符串有多少个。也是比较入门的题。。

  • dp[i][j][S]表示长度i(在自动机上转移k步)且后缀状态为自动机上第j个结点且当前包含咒语集合为S的方案数
  • dp[0][0][0]=1
  • 还是用我为人人转移,AC自动机上的结点要多一个域表示这个结点所代表咒语前缀包含的咒语集合。
 #include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
int tn,ch[][],fail[],flag[];
void insert(char *s,int k){
int x=;
for(int i=; s[i]; ++i){
int y=s[i]-'a';
if(ch[x][y]==) ch[x][y]=++tn;
x=ch[x][y];
}
flag[x]|=<<k;
}
void init(){
memset(fail,,sizeof(fail));
queue<int> que;
for(int i=; i<; ++i){
if(ch[][i]) que.push(ch[][i]);
}
while(!que.empty()){
int x=que.front(); que.pop();
for(int i=; i<; ++i){
if(ch[x][i]) que.push(ch[x][i]),fail[ch[x][i]]=ch[fail[x]][i],flag[ch[x][i]]|=flag[ch[fail[x]][i]];
else ch[x][i]=ch[fail[x]][i];
}
}
}
int getCnt(int s){
int res=;
for(int i=; i<; ++i){
if((s>>i)&) ++res;
}
return res;
}
int d[][][<<];
int main(){
char str[];
int n,m,k;
while(~scanf("%d%d%d",&n,&m,&k) && (n||m||k)){
tn=;
memset(ch,,sizeof(ch));
memset(flag,,sizeof(flag));
for(int i=; i<m; ++i){
scanf("%s",str);
insert(str,i);
}
init();
memset(d,,sizeof(d));
d[][][]=;
for(int i=; i<n; ++i){
for(int j=; j<=tn; ++j){
for(int k=; k<(<<m); ++k){
if(d[i][j][k]==) continue;
for(int y=; y<; ++y){
d[i+][ch[j][y]][k|flag[ch[j][y]]]+=d[i][j][k];
d[i+][ch[j][y]][k|flag[ch[j][y]]]%=;
}
}
}
}
int res=;
for(int i=; i<=tn; ++i){
for(int j=; j<(<<m); ++j){
if(getCnt(j)<k) continue;
res+=d[n][i][j];
res%=;
}
}
printf("%d\n",res);
}
return ;
}
上一篇:背水一战 Windows 10 (64) - 控件(WebView): 加载指定 HttpMethod 的请求, 自定义请求的 http header, app 与 js 的交互


下一篇:Android Cocos2d-x游戏集成友盟社会化组件分享功能