hdu 2825 Wireless Password AC自动机DP

6 篇文章 0 订阅
4 篇文章 0 订阅

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2825

发现只奥树上节点较少并且有最优子结构的问题就可以用ac自动机dp求解。

新发现了一点:

buildfail之后,之前等于-1的next值会赋值为与根节点相同分支的next值。

这是一个ac自动机上的状压dp,dp【i】【j】【k】代表用i个字符到达了节点j,并且当前覆盖了字符串集合k时有多少种情况。

在建立失败指针的时候,每个节点代表的集合值要或上其fail节点的集合值。即flag[i]|=flag[fail[i]]

根据ac自动机转移求解,最后找出所有覆盖集合>=k的dp[n][j][k]

最近感觉右手无力握紧会不自主颤抖,好难受

#include<bits/stdc++.h>
#define mod 20090717

using namespace std;


const int maxn=30+10;///多个模式串长度
const int N=30+10;///文章长度
const int lettersize=26;///看种类


char s[maxn];
char no[N];
long long dp[30][110][1030];

int pin(int i)
{
    int ans=0;
    while(i)
    {
        if(i&1)
            ans++;
        i>>=1;
    }
    return ans;
}

const int MAX=1e6+10;
struct Trie
{
    int next[MAX][lettersize],fail[MAX],end[MAX];
    int flag[MAX];
    int root,L;
    int newnode()
    {
        for (int i=0;i<lettersize;i++)
            next[L][i]=-1;
        end[L++]=0;
        return L-1;
    }
    void init()
    {
        L=0;
        root=newnode();
        memset(dp,0,sizeof dp);
        memset(flag,0,sizeof flag);
    }
    void insert(char *buf,int x)
    {
        int len=strlen(buf);
        int now=root;
        for (int i=0;i<len;i++)
        {
            if (next[now][buf[i]-'a']==-1)
                next[now][buf[i]-'a']=newnode();
            now=next[now][buf[i]-'a'];
        }
        end[now]++;
        flag[now]=1<<x;
    }
    void build ()
    {
        queue<int >Q;
        fail[root]=root;
        for (int i=0;i<lettersize;i++)
            if (next[root][i]==-1)
                next[root][i]=root;
            else
            {
                fail[next[root][i]]=root;
                Q.push(next[root][i]);
            }
        while (!Q.empty())
        {
            int now=Q.front();
            Q.pop();

            flag[now]=flag[now]|flag[fail[now]];

            for (int i=0;i<lettersize;i++)
                if (next[now][i]==-1)
                    next[now][i]=next[fail[now]][i];
                else
                {
                    fail[next[now][i]]=next[fail[now]][i];
                    Q.push(next[now][i]);
                }
        }
    }
}ac;


int main()
{
    int n,m,kk;
    while(~scanf("%d%d%d",&n,&m,&kk))
    {
//        printf("%d\n",pin(3));
        if(n==m&&m==kk&&n==0)
            break;
            ac.init();
        for(int i=0;i<m;i++)
                scanf("%s",s),ac.insert(s,i);
        ac.build();
//        for(int i=0;i<26;i++)
//            printf("next[4][%c]=%d\n",'a'+i,ac.next[5][i]);

//        for(int i=0;i<ac.L;i++)
//            printf("flag%d=%d\n",i,ac.flag[i]);
        dp[0][0][0]=1;
        for(int i=1;i<=n;i++)
        {
            for(int j=0;j<ac.L;j++)
            {
                for(int k=0;k<(1<<m);k++)
                {
                    if(dp[i-1][j][k])
                    {
                        for(int l=0;l<26;l++)
                        {
                            int son=ac.next[j][l];
                            int flag=k|ac.flag[son];
//                            if(ac.end[son]==1)
//                                dp[i][0][flag]=(dp[i-1][j][k]+dp[i][0][flag])%mod;

                                dp[i][son][flag]=(dp[i-1][j][k]+dp[i][son][flag])%mod;
//
//                            printf("dp %d %d %d=%d\n",i,son,flag,dp[i][son][flag]);
                            if(i==6&&son==6 &&flag==1)
                                printf("%d %d %d %lld %lld\n",j,ac.end[son],k,dp[i-1][j][k],dp[6][6][1]);
                        }
                    }
                }
            }
        }
         long long ans=0;
//for(int i=1;i<=n;i++)
//            for(int j=0;j<ac.L;j++)
//                for(int k=0;k<(1<<m);k++)
//                    printf("dp %d %d %d =%lld\n",i,j,k,dp[i][j][k]);
            for(int j=0;j<ac.L;j++)
            {
                for(int k=0;k<(1<<m);k++)
                    if(pin(k)>=kk)
                        ans=(ans+dp[n][j][k])%mod;
            }
            printf("%lld\n",ans);
    }
    return 0;
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值