zoukankan      html  css  js  c++  java
  • Trie图(模板)

    Trie图(蒟蒻听说AC自动机能做的题Trie图都能做,而且AC自动机可能被卡,就没学过AC自动机),最近想捡一捡,好久之前做的了。

    Trie图,就是一个在Trie树上建的图  大概描述一下

    比如说有几个字符串:

    abc

    abcd

    bcd

    bacd

    jdr

    ac

    先把它们存在Trie树中:

     就像KMP那样,做出这样的逻辑判断:

    bacd比较到第三位bac结果没有d,但起码bac有了,所以以bac为前缀的或以bac后缀为前缀的串是不用再比较前缀了。

    所以出现了fail指针,为失配情况重新定位方案。

    类似于next数组。

    无解(定位不到失配后新方案),就指向根表示无解。

    显而易见,首字母失配是一定没有方案的。

    其次,一个字母失配可以定位到父节点失配的自己值子节点。

    一个优化:没有新子节点的节点直接指向fail指针自己值子节点。

    建出来Trie图是这样的:

    匹配时类似KMP:

    模板代码(luogu P3808 【模板】AC自动机(简单版)):

     1 #include<queue>
     2 #include<cstdio>
     3 #include<cstring>
     4 #include<algorithm>
     5 struct trnt{
     6     int ch[26];
     7     int val;
     8     int fl;
     9 }tr[1000000];
    10 std::queue<int>Q;
    11 char tmp[1000003];
    12 int siz;
    13 int n;
    14 void add(char *a)
    15 {
    16     int len=strlen(a+1);
    17     int root=0;
    18     for(int i=1;i<=len;i++)
    19     {
    20         int c=a[i]-'a';
    21         if(!tr[root].ch[c])
    22             tr[root].ch[c]=++siz;
    23         root=tr[root].ch[c];
    24     }
    25     tr[root].val++;
    26 }
    27 void Build()
    28 {
    29     int root=0;
    30     for(int i=0;i<26;i++)
    31         if(tr[root].ch[i])
    32             Q.push(tr[root].ch[i]);
    33     while(!Q.empty())
    34     {
    35         root=Q.front();
    36         Q.pop();
    37         for(int i=0;i<26;i++)
    38         {
    39             if(tr[root].ch[i])
    40             {
    41                 tr[tr[root].ch[i]].fl=tr[tr[root].fl].ch[i];
    42                 Q.push(tr[root].ch[i]);
    43             }else
    44                 tr[root].ch[i]=tr[tr[root].fl].ch[i];
    45         }
    46     }
    47     return ;
    48 }
    49 int Cal(char *a)
    50 {
    51     int len=strlen(a+1);
    52     int ans=0;
    53     int root=0;
    54     for(int i=1;i<=len;i++)
    55     {
    56         int c=a[i]-'a';
    57         root=tr[root].ch[c];
    58         for(int j=root;j&&(tr[j].val!=-1);j=tr[j].fl)
    59         {
    60             ans+=tr[j].val;
    61             tr[j].val=-1;
    62         }
    63     }
    64     return ans;
    65 }
    66 int main()
    67 {
    68     scanf("%d",&n);
    69     for(int i=1;i<=n;i++)
    70     {
    71         scanf("%s",tmp+1);
    72         add(tmp);
    73     }
    74     Build();
    75     scanf("%s",tmp+1);
    76     printf("%d
    ",Cal(tmp));
    77     return 0;
    78 }
  • 相关阅读:
    POJ 1061 青蛙的约会(扩展欧几里得)
    贝祖定理(裴蜀定理)
    C语言 gets()和scanf()函数的区别
    非递归方式遍历二叉树
    zip包的解压
    八大基础排序中(直接插入排序,希尔排序,冒泡排序, 快速排序,归并排序,简单选择排序)
    数字反序与数字的和
    合并两个有序数组,合并后数组仍有序
    使用递归方式和非递归方式求斐波那契数
    求100到999之内的水仙花数
  • 原文地址:https://www.cnblogs.com/blog-Dr-J/p/9670219.html
Copyright © 2011-2022 走看看