【字符串新武器】后缀自动机

2024-08-24 11:48

本文主要是介绍【字符串新武器】后缀自动机,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

发链:http://neroysq.blogcn.com/articles/%E5%90%8E%E7%BC%80%E8%87%AA%E5%8A%A8%E6%9C%BA%E5%88%9D%E6%8E%A2.html

http://blog.sina.com.cn/s/blog_7812e98601012cim.html

详细构造见上述链接,此处介绍性质与理解

后缀自动机具有两大性质,考虑转移边为自动机,考虑父亲边为逆序后缀树(后缀数组为后缀树的dfs序),如果忽略字符集大小的常数,构造复杂度为o(n)

很好很强大,更深的性质呢?

对于一个节点i,根s

1、整个自动机可以接受字符串的所有子串,且自动机为一个有向拓扑图

2、一条到i的路径唯一对应一个子串

3、所有i可以接受的状态互相成后缀关系,且长度连续

4、对于i的父亲rt[i],rt[i]所能接受的所有子串都是i所能接受的所有子串的后缀

构造后缀自动机的时候有一步是判断l[a]+1==l[b],为的是防止多余状态被自动机接受,如果l[a]+1==l[b],那么b所能接受的最长子串是合法的,根据3,4可知所有其他状态也都是合法的,否则l[a]+1!=l[b]可能会出现不合法状态,那么新建节点使l[c]=l[a]+1就可以筛出合法状态,而rt指针的修改可以看作是逆序后缀树的压缩边被解压缩出一个节点并添加一个新的叶子节点(即新逆序后缀)

5、统计方式的理解。从后续题目来看有两种统计方法,一是根据转移边dp,二是根据父亲边dp,其实是统计两个不同东西,转移边dp可以统计出有多少子串以此节点接受的状态为前缀,根据父亲边dp可以统计出以该节点接受的最长子串为后缀的子串有多少(逆序后缀树i所代表的是i所接受最长子串)

6、其实字典树也可以建立后缀自动机,考虑我们对于串建的时候,记last是为了找出最后一个的最长子串,而在字典树上就是字典树的父亲,所以按dfs序及父亲就可以建出字典树的后缀自动机

7、(逆序后缀树i所代表的是i所接受最长子串)这句话的意思是,将该串所有的逆序后缀建立一棵后缀树,而自动机中的每个节点i,它在后缀树中一直沿父亲边走向根\所代表的逆序后缀==节点i在自动机中接受到的最长子串

总而言之,后缀自动机兼有两种特性,如果想知道后缀的公共前缀之类,或子串重复次数等等与后缀联系的上的,可以用后缀树的性质,如果与所有子串相关,自动机就比较好用

1.SPOJ NSUBSTR

题意:给一个字符串S,令F(x)表示S的所有长度为x的子串中,出现次数的最大值。求F(1)..F(Length(S)) (感谢clj的翻译>_<)

用逆序后缀树的话叶子数就是出现次数(可以基排后直接统计,那时理解还不是很深,所以真的把树建出来了)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. int tail[500001],net[2000000],sora[2000000],len[500001],f[500001],rt[500001];  
  5. int next[500001][27],ans[500001],s1,ss;  
  6. char ch[500001];  
  7. void dfs(int x)  
  8. {  
  9.     int i,ne;  
  10.     f[x]=(x==tail[x]) ? 1 : 0;  
  11.     for (i=x;net[i]!=0;) {  
  12.     i=net[i],ne=sora[i];  
  13.     if (ne==rt[x]) continue;  
  14.     dfs(ne);  
  15.     f[x]+=f[ne];  
  16.     }  
  17.     if (f[x]>ans[len[x]]) ans[len[x]]=f[x];  
  18. }  
  19. void origin() {for (int i=1;i<=s1;i++) tail[i]=i;ss=s1;}  
  20. void link(int x,int y)   
  21. {  
  22.     ss++,net[tail[x]]=ss,tail[x]=ss,sora[ss]=y;  
  23. }  
  24. void init()  
  25. {  
  26.     int l,i,ne,last,x,y,j;  
  27.     scanf("%s\n",ch+1);  
  28.     l=strlen(ch+1);ch[0]='z'+1;  
  29.     next[0][ch[0]-'a']=++s1,rt[s1]=0,len[s1]=1,last=s1;  
  30.     for (i=1;i<=l;i++) {  
  31.     ne=ch[i]-'a';++s1;len[s1]=i+1;  
  32.     for (x=last,last=s1;!next[x][ne]&& x;x=rt[x]) next[x][ne]=s1;  
  33.     y=next[x][ne];  
  34.     if (!y) {next[x][ne]=s1;continue;}  
  35.     if (len[x]+1==len[y]) rt[s1]=y;  
  36.     else {  
  37.         s1++;len[s1]=len[x]+1;  
  38.         for (j=0;j<=26;j++) next[s1][j]=next[y][j];  
  39.         rt[last]=s1;rt[s1]=rt[y],rt[y]=s1;  
  40.         for (;x;x=rt[x]) if (next[x][ne]==y) next[x][ne]=s1;else break;  
  41.         if (next[x][ne]==y) next[x][ne]=s1;  
  42.     }  
  43.     }  
  44.     origin();  
  45.     for (i=1;i<=s1;i++) link(rt[i],i);  
  46.     dfs(0);  
  47.     for (i=1;i<=l;i++) printf("%d\n",ans[i]);  
  48. }  
  49. int main()  
  50. {  
  51.     freopen("spoj8222.in","r",stdin);  
  52.     freopen("spoj8222.out","w",stdout);  
  53.      init();  
  54.     return 0;  
  55. }  


2.SPOJ LCS

题意:求两个字符串的最长公共子串

由于后缀自动机能接受所有子串,所以对其中一个建好后缀自动机后,用另一个在自动机上匹配(匹配失败可以由rt指针找出最长公共后缀子串),就可以求出任意位置能匹配的最长长度

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. int ans,sum,len[500000],next[500000][26],rt[500000],last,s1;  
  5. char ch[500000];  
  6. void init()  
  7. {  
  8.     int i,j,l,s,ne,x,y;  
  9.     scanf("%s\n",ch+1);  
  10.     l=strlen(ch+1);  
  11.     for (i=1,last=0;i<=l;i++) {  
  12.     ne=ch[i]-'a';  
  13.     for (x=last,last=++s1;x && !next[x][ne];x=rt[x]) next[x][ne]=s1;  
  14.     y=next[x][ne];len[s1]=i;  
  15.     if (!y) {next[x][ne]=s1;continue;}  
  16.     else {  
  17.         if (len[x]+1==len[y]) rt[s1]=y;  
  18.         else {  
  19.         len[++s1]=len[x]+1;  
  20.         for (j=0;j<='z'-'a';j++) next[s1][j]=next[y][j];  
  21.         rt[last]=s1,rt[s1]=rt[y],rt[y]=s1;  
  22.         for (;x && next[x][ne]==y;x=rt[x]) next[x][ne]=s1;  
  23.         if (next[x][ne]==y) next[x][ne]=s1;  
  24.         }  
  25.     }  
  26.     }  
  27.     scanf("%s\n",ch+1);  
  28.     l=strlen(ch+1);  
  29.     for (s=0,i=1,ans=sum=0;i<=l;i++) {  
  30.     ne=ch[i]-'a';  
  31.     for (;s && !next[s][ne];sum=len[s=rt[s]]) ;  
  32.     if (next[s][ne]) sum++,s=next[s][ne];  
  33.     if (sum>ans) ans=sum;  
  34.     }  
  35.     printf("%d\n",ans);  
  36. }  
  37. int main()  
  38. {  
  39.     freopen("spoj1811.in","r",stdin);  
  40.     freopen("spoj1811.out","w",stdout);  
  41.      init();  
  42.     return 0;  
  43. }  


3.SPOJ LCS2

题意:求n个字符串的最长公共子串(n<=10)

按9个字符串建的话,ldl超时了,所以不妨按一个建,用其余9个在上面匹配,注意如果一个点的某子串被匹配了,那他父亲代表的所有子串都会被匹配,因为满足后缀关系(参见之前性质分析)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. int f[200001][10],rt[200001],next[200001][26],st[200001],len[200001],g[200001],ws[200001];  
  5. int s1,last,ans,sum,tim;  
  6. char ch[100001];  
  7. void init()  
  8. {  
  9.     int i,l,j,x,y,ne,s;  
  10.     scanf("%s\n",ch+1);  
  11.     l=strlen(ch+1);  
  12.     for (last=0,i=1;i<=l;i++) {  
  13.     ne=ch[i]-'a';  
  14.     for (x=last,last=++s1;!next[x][ne] && x;x=rt[x]) next[x][ne]=s1;  
  15.     y=next[x][ne],len[s1]=i;  
  16.     if (!y) {next[x][ne]=s1;continue;}  
  17.     else {  
  18.         if (len[x]+1==len[y]) rt[s1]=y;  
  19.         else {  
  20.         len[++s1]=len[x]+1;  
  21.         for (j=0;j<='z'-'a';j++) next[s1][j]=next[y][j];  
  22.         rt[last]=s1,rt[s1]=rt[y],rt[y]=s1;  
  23.         for (;x && next[x][ne]==y;x=rt[x]) next[x][ne]=s1;  
  24.         if (next[x][ne]==y) next[x][ne]=s1;  
  25.         }  
  26.     }  
  27.     }  
  28.     for (i=1;i<=l;i++) ws[i]=0;  
  29.     for (i=1;i<=s1;i++) ws[g[i]=len[i]]++;  
  30.     for (i=1;i<=l;i++) ws[i]+=ws[i-1];  
  31.     for (i=s1;i>=1;i--) st[ws[len[i]]--]=i;  
  32.     for (;!(scanf("%s\n",ch+1)==EOF);) {  
  33.     l=strlen(ch+1);  
  34.     tim++;  
  35.     for (s=0,sum=0,i=1;i<=l;i++) {  
  36.         ne=ch[i]-'a';  
  37.         for (;s && !next[s][ne];sum=len[s=rt[s]]) ;  
  38.         if (next[s][ne]) sum++,s=next[s][ne];  
  39.         f[s][tim]=(sum>f[s][tim]) ? sum : f[s][tim];  
  40.     }  
  41.     for (i=s1;i>=1;i--) {  
  42.         s=st[i];  
  43.         f[rt[s]][tim]=(f[rt[s]][tim]>f[s][tim]) ? f[rt[s]][tim] : f[s][tim];  
  44.         g[s]=(g[s]<f[s][tim]) ? g[s] : f[s][tim];  
  45.     }  
  46.     }  
  47.     ans=0;  
  48.     for (i=1;i<=s1;i++) ans=(g[st[i]]>ans) ? g[st[i]] : ans;  
  49.     printf("%d\n",ans);  
  50. }  
  51. int main()  
  52. {  
  53.   // freopen("spoj1812.in","r",stdin);  
  54.   // freopen("spoj1812.out","w",stdout);  
  55.      init();  
  56.     return 0;  
  57. }  


4.SPOJ SUBLEX

题意:给定一个字符串,用它的所有子串建立一棵字母树,每次找第k小的子串。

自动机可以接受所有子串,如果我们知道一个节点能拓展出多少状态(实际是可由dp完成),那么直接按顺序在自动机上走,就可以o(n)的回答询问(实际是26n,spoj上会超时,要用hash将转移边优化一下)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. int f[180001],st[180001],len[180001],rt[180001],next[180001][26],ws[180001],net[500000],tail[180001],sora[500000],pow[500000];  
  5. int m,last,s1,ss;  
  6. char ch[180001];  
  7. void link(int x,int y,int i)  
  8. {  
  9.     ss++,net[tail[x]]=ss,tail[x]=ss,sora[ss]=y,pow[ss]=i;  
  10. }  
  11. void init()  
  12. {  
  13.     int i,j,l,x,y,ne,s,k;  
  14.     scanf("%s\n",ch+1);  
  15.     l=strlen(ch+1);  
  16.     for (last=0,i=1;i<=l;i++) {  
  17.     ne=ch[i]-'a';  
  18.     for (x=last,last=++s1;x && !next[x][ne];x=rt[x]) next[x][ne]=s1;  
  19.     y=next[x][ne],len[s1]=i;  
  20.     if (!y) {next[x][ne]=s1;continue;}   
  21.     else {  
  22.         if (len[x]+1==len[y]) rt[s1]=y;  
  23.         else {  
  24.         len[++s1]=len[x]+1;  
  25.         for (j=0;j<='z'-'a';j++) next[s1][j]=next[y][j];  
  26.         rt[last]=s1,rt[s1]=rt[y],rt[y]=s1;  
  27.         for (;x && next[x][ne]==y;x=rt[x]) next[x][ne]=s1;  
  28.         if (next[x][ne]==y) next[x][ne]=s1;  
  29.         }  
  30.     }  
  31.     }  
  32.     for (i=1;i<=s1;i++) ws[len[i]]++,tail[i]=i;ss=s1;  
  33.     for (i=1;i<=l;i++) ws[i]+=ws[i-1];  
  34.     for (i=s1;i>=1;i--) st[ws[len[i]]--]=i;  
  35.     for (i=s1;i>=0;i--) {  
  36.     x=st[i],f[x]++;  
  37.     for (j=0;j<='z'-'a';j++) {  
  38.         if (!next[x][j]) continue;  
  39.         f[x]+=f[next[x][j]];  
  40.         link(x,next[x][j],j);  
  41.     }  
  42.     }  
  43.     scanf("%d\n",&m);  
  44.     for (;m;m--) {  
  45.     scanf("%d\n",&k);  
  46.     for (s=0;k;) {  
  47.         for (i=s;net[i]!=0;) {  
  48.         i=net[i],ne=sora[i];  
  49.         if (f[ne]<k) k-=f[ne];else break;  
  50.         }  
  51.         printf("%c",'a'+pow[i]),k--;  
  52.         s=sora[i];  
  53.     }  
  54.     printf("\n");  
  55.     }  
  56. }  
  57. int main()  
  58. {  
  59.     freopen("spoj7258.in","r",stdin);  
  60.     freopen("spoj7258.out","w",stdout);  
  61.      init();  
  62.     return 0;  
  63. }  


多串建后缀自动机,通常的方法是加入最小字符进行分割,但是其实不需要这么复杂,只需要每个串都重新从根节点开始插入就可以了,这样子会产生一些冗余节点(没有从根到它的路径),但是能遍历到的部分都是正确的,其实就相当于先建了一棵字母树,然后每个节点的last就是它在字母树上的祖先,然后依照这个字母树建后缀自动机

updata:不会产生冗余节点的做法(冗余节点在运用父亲边的时候可能会出问题),如果先建字母树,再建自动机是不会出问题的,但是不需要这么麻烦,只需边插边走的时候判断一下下一个节点是不是还在最长链上即可,这样就保证走的是字母树的边

hdu4436

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. #include <algorithm>  
  5. #include <iostream>  
  6. const int mo=2012;  
  7. using namespace std;  
  8. int ss,n,f[2000000],next[500000][10],l[2000000],rt[2000000],g[2000000];  
  9. char ch[2000000];  
  10. int u[500000];  
  11. void add(int &last,int chr)  
  12. {  
  13.     int x,y;  
  14.     ss++,l[ss]=l[last]+1;  
  15.     for (x=last,last=ss;x && (!next[x][chr]);x=rt[x]) next[x][chr]=ss;  
  16.     y=next[x][chr];  
  17.     if (!y) next[x][chr]=ss,rt[ss]=0;  
  18.     else if (l[x]+1==l[y]) rt[ss]=y;  
  19.     else {  
  20.         ++ss,l[ss]=l[x]+1;  
  21.         for (int j=0;j<=9;j++) next[ss][j]=next[y][j];  
  22.         rt[ss]=rt[y],rt[y]=ss,rt[last]=ss;  
  23.         for (;x && (next[x][chr]==y);x=rt[x]) next[x][chr]=ss;  
  24.         if (next[x][chr]==y) next[x][chr]=ss;  
  25.     }  
  26. }  
  27. void origin()  
  28. {  
  29.     for (int i=0;i<=ss;i++) {  
  30.         l[i]=rt[i]=0;  
  31.         for (int j=0;j<=9;j++)  
  32.             next[i][j]=0;  
  33.     }  
  34.     ss=0;  
  35. }  
  36. bool cmp(int i,int j)  
  37. {  
  38.     return l[i]<l[j];  
  39. }  
  40. int main()  
  41. {  
  42.     freopen("input.txt","r",stdin);  
  43.     //freopen("output.txt","w",stdout);  
  44.     for (;scanf("%d",&n)==1;) {  
  45.         origin();  
  46.         for (int i=1;i<=n;i++) {  
  47.             scanf("%s",ch+1);  
  48.             int len=strlen(ch+1);  
  49.             int s=0;  
  50.             for (int i=1;i<=len;i++) {  
  51.                 int chr=ch[i]-'0';  
  52.                 add(s,chr);  
  53.             }  
  54.         }  
  55.         for (int i=1;i<=ss;i++) u[i]=i,f[i]=0,g[i]=0;  
  56.         sort(u+1,u+ss+1,cmp);  
  57.         int ans=0;  
  58.         for (int i=1;i<=9;i++)   
  59.             if (next[0][i]) f[next[0][i]]=i,g[next[0][i]]=1;  
  60.         for (int i=1;i<=ss;i++) {  
  61.             int ne=u[i];  
  62.             ans=(ans+f[ne])%mo;  
  63.             for (int j=0;j<=9;j++) {  
  64.                 int na=next[ne][j];  
  65.                 if (na)  
  66.                     f[na]=(f[na]+f[ne]*10+g[ne]*j)%mo,g[na]=(g[na]+g[ne])%mo;  
  67.             }  
  68.         }  
  69.         printf("%d\n",ans);  
  70.     }  
  71.     return 0;  
  72. }   


hdu4436(updata版)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <cstdio>  
  2. #include <cstdlib>  
  3. #include <cstring>  
  4. #include <algorithm>  
  5. #include <iostream>  
  6. const int mo=2012;  
  7. using namespace std;  
  8. int ss,n,f[2000000],next[500000][10],l[2000000],rt[2000000],g[2000000];  
  9. char ch[2000000];  
  10. int u[500000];  
  11. void add(int &last,int chr)  
  12. {  
  13.     int x,y;  
  14.     ss++,l[ss]=l[last]+1;  
  15.     for (x=last,last=ss;x && (!next[x][chr]);x=rt[x]) next[x][chr]=ss;  
  16.     y=next[x][chr];  
  17.     if (!y) next[x][chr]=ss,rt[ss]=0;  
  18.     else if (l[x]+1==l[y]) rt[ss]=y;  
  19.     else {  
  20.         ++ss,l[ss]=l[x]+1;  
  21.         for (int j=0;j<=9;j++) next[ss][j]=next[y][j];  
  22.         rt[ss]=rt[y],rt[y]=ss,rt[last]=ss;  
  23.         for (;x && (next[x][chr]==y);x=rt[x]) next[x][chr]=ss;  
  24.         if (next[x][chr]==y) next[x][chr]=ss;  
  25.     }  
  26. }  
  27. void origin()  
  28. {  
  29.     for (int i=0;i<=ss;i++) {  
  30.         l[i]=rt[i]=0;  
  31.         for (int j=0;j<=9;j++)  
  32.             next[i][j]=0;  
  33.     }  
  34.     ss=0;  
  35. }  
  36. bool cmp(int i,int j)  
  37. {  
  38.     return l[i]<l[j];  
  39. }  
  40. int main()  
  41. {  
  42.     freopen("input.txt","r",stdin);  
  43.     freopen("output.txt","w",stdout);  
  44.     for (;scanf("%d",&n)==1;) {  
  45.         origin();  
  46.         for (int i=1;i<=n;i++) {  
  47.             scanf("%s",ch+1);  
  48.             int len=strlen(ch+1);  
  49.             int s=0;  
  50.             for (int i=1;i<=len;i++) {  
  51.                 int chr=ch[i]-'0';  
  52.                 int ne=next[s][chr];  
  53.                 if (!ne || (l[s]+1!=l[ne])) add(s,chr);  
  54.                 else s=ne;  
  55.             }  
  56.         }  
  57.         for (int i=1;i<=ss;i++) u[i]=i,f[i]=0,g[i]=0;  
  58.         sort(u+1,u+ss+1,cmp);  
  59.         int ans=0;  
  60.         for (int i=1;i<=9;i++)   
  61.             if (next[0][i]) f[next[0][i]]=i,g[next[0][i]]=1;  
  62.         for (int i=1;i<=ss;i++) {  
  63.             int ne=u[i];  
  64.             ans=(ans+f[ne])%mo;  
  65.             for (int j=0;j<=9;j++) {  
  66.                 int na=next[ne][j];  
  67.                 if (na)  
  68.                     f[na]=(f[na]+f[ne]*10+g[ne]*j)%mo,g[na]=(g[na]+g[ne])%mo;  
  69.             }  
  70.         }  
  71.         printf("%d\n",ans);  
  72.     }  
  73.     return 0;  
  74. }   

这篇关于【字符串新武器】后缀自动机的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102392

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

Python在固定文件夹批量创建固定后缀的文件(方法详解)

《Python在固定文件夹批量创建固定后缀的文件(方法详解)》文章讲述了如何使用Python批量创建后缀为.md的文件夹,生成100个,代码中需要修改的路径、前缀和后缀名,并提供了注意事项和代码示例,... 目录1. python需求的任务2. Python代码的实现3. 代码修改的位置4. 运行结果5.

python修改字符串值的三种方法

《python修改字符串值的三种方法》本文主要介绍了python修改字符串值的三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录第一种方法:第二种方法:第三种方法:在python中,字符串对象是不可变类型,所以我们没办法直接

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

C#中字符串分割的多种方式

《C#中字符串分割的多种方式》在C#编程语言中,字符串处理是日常开发中不可或缺的一部分,字符串分割是处理文本数据时常用的操作,它允许我们将一个长字符串分解成多个子字符串,本文给大家介绍了C#中字符串分... 目录1. 使用 string.Split2. 使用正则表达式 (Regex.Split)3. 使用

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

hdu 3065 AC自动机 匹配串编号以及出现次数

题意: 仍旧是天朝语题。 Input 第一行,一个整数N(1<=N<=1000),表示病毒特征码的个数。 接下来N行,每行表示一个病毒特征码,特征码字符串长度在1—50之间,并且只包含“英文大写字符”。任意两个病毒特征码,不会完全相同。 在这之后一行,表示“万恶之源”网站源码,源码字符串长度在2000000之内。字符串中字符都是ASCII码可见字符(不包括回车)。

POJ 1625 自动机

给出包含n个可见字符的字符集,以下所提字符串均由该字符集中的字符构成。给出p个长度不超过10的字符串,求长为m且不包含上述p个字符串的字符串有多少个。 g++提交 int mat[108][108] ;int matn ;int N ;map<char ,int> to ;//ACconst int maxm = 108 ;const int kin

zoj 3228 ac自动机

给出一个字符串和若干个单词,问这些单词在字符串里面出现了多少次。单词前面为0表示这个单词可重叠出现,1为不可重叠出现。 Sample Input ab 2 0 ab 1 ab abababac 2 0 aba 1 aba abcdefghijklmnopqrstuvwxyz 3 0 abc 1 def 1 jmn Sample Output Case 1 1 1 Case 2

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数