zoukankan      html  css  js  c++  java
  • [JSOI2007]文本生成器

    问题描述

    对于一个长度为 \(n\) 的串 \(S\) ,有多少可能情况的串 \(S\) 使得 \(S\) 的子串中至少包含一个给定的串,给定的串有 \(m\)

    解法

    由多模式串匹配想到AC自动机,由计数想到dp

    首先建好Trie图,更新所有end标记。记 \(dp[now][st][flag]\) 表示当前正在匹配第 \(st\) 位,已确定的串匹配到了Trie图上的 \(now\) 号节点,\(flag\) 表示是否已经出现过给定的模式串。总共有 \((\sum_{i=1}^{m} |s_i|)\times n \times 2\) 种状态,显然要记忆化搜索。对于每个状态枚举下一位所有可能的 \(26\) 位字符,并在Trie图上进行节点转移,flag或上end标记,最后累加答案。当 \(st=n+1\) 即已经匹配完成时,返回值为flag的状态(显然若有给定字符串则贡献 \(1\) 的答案)。最终的结果为初始状态 \(dp[0][1][false]\)

    #include<stdio.h>
    #include<string.h>
    #include<queue>
    using namespace std;
    #define Mod 10007
    #define N 107
    
    template<class T>
    inline void read(T &x){
        x=0;char c=getchar();T flag=1;
        while(c<'0'||c>'9'){if(c=='-')flag=-1;c=getchar();}
        while(c>='0'&&c<='9'){x=(x<<1)+(x<<3)+c-48;c=getchar();}
        x*=flag;
    }
    
    struct Trie{
        int vis[26],fail,end;
    }go[N*100];
    
    int cnt=0,n,m,f[N*100][N][2];
    char s[N];
    queue<int> Q;
    
    inline void insert(){
        int now=0,len=strlen(s);
        for(int i=0;i<len;i++){
            int to=s[i]-'A';
            if(!go[now].vis[to])
                go[now].vis[to]=++cnt;
            now=go[now].vis[to];
        }
        go[now].end=1;
    }
    inline void Get_fail(){
        for(int i=0;i<26;i++)
            if(go[0].vis[i]) Q.push(go[0].vis[i]);
        while(!Q.empty()){
            int u=Q.front();Q.pop();
            for(int i=0;i<26;i++)
                if(go[u].vis[i]){
                    go[go[u].vis[i]].fail=go[go[u].fail].vis[i];
                    go[go[u].vis[i]].end|=go[go[go[u].fail].vis[i]].end;
                    Q.push(go[u].vis[i]);
                }else go[u].vis[i]=go[go[u].fail].vis[i];
        }
    }
    int dfs(int now,int st,bool flag){
        if(st==m+1) return flag;
        if(f[now][st][flag]!=-1) return f[now][st][flag];
        f[now][st][flag]=0;int ret=0;
        for(int i=0;i<26;i++){
            int to=go[now].vis[i];
            ret=(ret+dfs(to,st+1,flag|go[to].end))%Mod;
        }
        return f[now][st][flag]=ret;
    }
    int main(){
    //    freopen("dict.in","r",stdin);
    //    freopen("dict.out","w",stdout);
        memset(f,-1,sizeof(f));
        read(n);read(m);
        for(int i=1;i<=n;i++){
            scanf("%s",s);
            insert();
        }
        Get_fail();
        printf("%d",dfs(0,1,false));
    }
    /*
    2 2
    AB
    BA
    */
    
  • 相关阅读:
    mysql存储过程之游标
    ip后面带端口号如何做域名解析
    将博客搬至CSDN
    java微信公众号JSAPI支付以及所遇到的坑
    button元素的id与onclick的函数名字相同 导致方法失效的问题
    在centOS使用systemctl配置启动多个tomcat
    mysql正则表达式,实现多个字段匹配多个like模糊查询
    web前端基础知识-(二)CSS基本操作
    web前端基础知识-(一)html基本操作
    python学习笔记-(十六)python操作mysql
  • 原文地址:https://www.cnblogs.com/wwlwQWQ/p/13550675.html
Copyright © 2011-2022 走看看