(いくつかのテンプレート文字列AC +オートマトンのテキスト文字列を登場)HDOJ-2896

ウイルス

HDOJ-2896

  • AC自動機は主に、解決するために使用された回数に表示される各テンプレート文字列を記録するために使用されるクエリ機能の変化、続く、とレコード番号のテンプレート文字列に関数を挿入されます
  • なお、最良の構造の使用ではなく、それ以外の場合は1回の使用のmemsetをアウト時間
  • 何AC出力制限がなかった最後の時間は問題を超え、私は後でNUM開口のアレイが小さすぎる見出され、そして唯一505は、実際には、N個のスペースを必要とします。
  • もう一つの問題は、私が以前にMLE簡単に提出したときに、メモリが成果に、私はAC機械が自動的に構造体にアーカイブに置くまでは、オーバーラン、およびノー​​ドmemsettrieツリーを使用して起動しませんでしたが、ということですmemsetの上のノードは、スペースを節約するノードを構築する場合。
//AC自动机,复杂度为O(|t|+m),t表示文本串的长度,m表示模板串的个数
#include<iostream>
#include<cstring>
#include<string>
#include<cstdio>
#include<algorithm>
#include<queue>
#include<set>
using namespace std;
const int N=56600;
int n;
string s[502];
set<int>::iterator it;
string t;
struct ACM{
    int tree[N][128];//trie树上的结点,tree[i][j]表示i结点后面加一条j的边所对应的的结点
    int total;//总结点
    int num[N];//num[i]表示结点i上对应的模板串的编号
    int fail[N];//失配指针,fail[i]指向所有模板串的前缀中匹配当前状态的最长后缀,指向的是最长后缀(和当前状态的后缀是匹配的,即相同,不过要最长)
    int val[N];//val[j]表示结点j所对应的模板串在文本串中出现的次数
    set<int> se;
    queue<int> q;
    void init(){
        total=0;
        val[0]=fail[0]=num[0]=0;
        memset(tree[0],0,sizeof(tree[0]));
    }
    int idx(char c){//用来求字符c对应的编号(0-25)
        return c-0;
    }
    void insert(string s,int id){//类似于后缀树的插入一个模板串.id表示所有模板串中该模板串的编号
        int u=0;
        for(int i=0;i<s.length();i++){
            if(!tree[u][idx(s[i])])
                tree[u][idx(s[i])]=++total;
            memset(tree[total],0,sizeof(0));
            u=tree[u][idx(s[i])];
        }
        num[u]=id;
    }
    void build(){//建AC自动机以及fail数组
        for(int i=0;i<128;i++){
            if(tree[0][i]){
                q.push(tree[0][i]);
                fail[tree[0][i]]=0;
            }
        }
        while(!q.empty()){
            int u=q.front();
            q.pop();
            for(int i=0;i<128;i++){
                if(tree[u][i]){//如果结点u连的边为i对应的结点存在,则将这个存在的结点的fail指针指向父节点失配指针指向的结点的连的边为i所对应的的结点
                    fail[tree[u][i]]=tree[fail[u]][i];
                    q.push(tree[u][i]);
                }else{//类似于状态压缩,不至于每次fail指针跳转很多次,只需每次跳转一次,相当于构建了图
                    tree[u][i]=tree[fail[u]][i];
                }
            }
        }
    }
    void query(string t){//s为要查找的文本串
        se.clear();
        int u=0;
        int res=0;//记录答案,所有的模板串共出现了多少次
        for(int i=0;i<t.length();i++){
            u=tree[u][idx(t[i])];
            for(int j=u;j>0;){
                val[j]++;
                j=fail[j];
            }
        }
        for(int i=0;i<=total;i++){//注意:这里是i<=total ,而不是<total,否则报错
            if(num[i]){//如果i结点对应着一个模板串,该模板串编号为num[i].
                if(val[i]>0){
                    se.insert(num[i]);//模板串对应的编号插入set中
                    val[i]=0;
                }
            }
        }
        //return res;
    }
};
ACM ac;
int main(){
    ios::sync_with_stdio(false);
    cin.tie(0);
    cin>>n;
    ac.init();
    for(int i=1;i<=n;i++){
        cin>>s[i];
        ac.insert(s[i],i);
    }
    ac.build();
    int m;
    cin>>m;
    int ans=0;
    for(int i=1;i<=m;i++){
        cin>>t;
        //memset(val,0,sizeof(val));
        ac.query(t);
        int size=ac.se.size();
        if(size>0){
            //cout<<se.size()<<endl;
            ans++;
            cout<<"web "<<i<<":";
            int cnt=0;
            for(it=ac.se.begin();it!=ac.se.end();it++){
                cout<<" "<<*it;
            }
            cout<<endl;
        }
    }
    cout<<"total: "<<ans<<endl;
    //system("pause");
    return 0;
}

おすすめ

転載: www.cnblogs.com/GarrettWale/p/11332697.html