bzoj4032: [HEOI2015]最短不公共子串(SAM+DP)

4032: [HEOI2015]最短不公共子串

题目:传送门 

 


 

 

题解:

   陈年老题良心%你赛膜爆嘎爷

   当初做题...一眼SAM...结果只会两种直接DP的情况...

   情况1: 直接设f[i][j] 表示的是a串的第i个位置和b串的第j个位置开始的最长公共前缀(灵感来源于SA)。然后就枚举开头直接瞎搞啊。。。

   情况2: 定义一个last[i][j] 表示b串的位置i后面的第一个字符j,贪心的思想直接做,匹配不到位置就直接记录答案了嘛

   情况3: 因为是在b串当中找子串,a中找子序列,那就考虑对b建SAM,然后定义da[] 表示的是a串的第i个位置结尾的串在b串中跑到第j个状态的最短长度(具体看代码)、

   情况4: 利用情况2预处理的last,贪心思想,仿照情况三把last当作自动机来跑

    


 

 

代码:

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<cstdlib>
 4 #include<cmath>
 5 #include<algorithm>
 6 #define inf 1e9 
 7 using namespace std;
 8 struct SAM
 9 {
10     int son[50];
11 }ch[5100];int cnt,dep[5100],a[5100],fail[5100],da[5100],la,root,ss;、
12 //da 表示的是a串的第i个位置结尾的串在b串中跑到第j个状态的最短长度 
13 void add(int k)
14 {
15     int x=a[k];
16     int p=la,np=++cnt;dep[np]=k;
17     while(p!=0 && ch[p].son[x]==0)ch[p].son[x]=np,p=fail[p];
18     if(p==0)fail[np]=root;
19     else
20     {
21         int q=ch[p].son[x];
22         if(dep[p]+1==dep[q])fail[np]=q;
23         else
24         {
25             int nq=++cnt;dep[nq]=dep[p]+1;
26             ch[nq]=ch[q];fail[nq]=fail[q];fail[np]=fail[q]=nq;
27             while(p && ch[p].son[x]==q)ch[p].son[x]=nq,p=fail[p];
28         }
29     }
30     la=np;
31 }
32 int len1,len2,opt;
33 int c[50],last[2100][50];//位置i后面的第一个字符j 
34 char sa[2100],sb[2100];
35 int f[2100][2100];//分别以i,j位置开头的最长公共前缀
36 int main()
37 {
38     //freopen("a.in","r",stdin);
39     //freopen("a.out","w",stdout);
40     //scanf("%d",&opt);
41     scanf("%s",sa+1);len1=strlen(sa+1);
42     scanf("%s",sb+1);len2=strlen(sb+1);
43     //if(opt==1)
44     //{
45         memset(f,0,sizeof(f));int ans=inf,len=0,maxx=0;
46         for(int i=len1;i>=1;i--)for(int j=len2;j>=1;j--)if(sa[i]==sb[j])f[i][j]=f[i+1][j+1]+1;
47         for(int i=1;i<=len1;i++)
48         {        
49             maxx=0;for(int j=1;j<=len2;j++)maxx=max(maxx,f[i][j]);
50             if(maxx!=len1-i+1)ans=min(ans,maxx+1);//加一定不相等 
51         }
52         if(ans==inf)printf("-1\n");
53         else printf("%d\n",ans);
54     //}
55     //else if(opt==2)
56     //{
57         len=0,ans=inf;for(int i=1;i<=50;i++)c[i]=inf;sb[0]='a';
58         for(int i=len2;i>=0;i--){for(int j=1;j<=26;j++)last[i][j]=c[j];c[sb[i]-'a'+1]=i;}
59         for(int i=1;i<=len1;i++)
60         {
61             len=0;
62             for(int j=i;j<=len1;j++)
63             {
64                 len=last[len][sa[j]-'a'+1];
65                 if(len>len1){ans=min(ans,j-i+1);break;}//跳出去证明没有了
66             }
67         }
68         if(ans==inf)printf("-1\n");
69         else printf("%d\n",ans);
70     //}
71 //    else if(opt==3)
72 //    {
73         cnt=0;root=la=++cnt;memset(da,63,sizeof(da));
74         for(int i=1;i<=len2;i++)a[i]=sb[i]-'a';for(int i=1;i<=len2;i++)add(i);ss=0;da[1]=0;ans=inf;
75         for(int i=1;i<=len1;i++)
76             for(int j=1;j<=cnt;j++)
77             {
78                 ss=ch[j].son[sa[i]-'a'];
79                 if(ss==0)ans=min(ans,da[j]+1);else da[ss]=min(da[ss],da[j]+1);
80             }
81         if(ans==inf)printf("-1\n");
82         else printf("%d\n",ans);
83 //    }
84     //else if(opt==4)
85     //{
86         ans=inf;//for(int i=1;i<=50;i++)c[i]=inf;sb[0]='a';
87         //for(int i=len2;i>=0;i--){for(int j=1;j<=26;j++)last[i][j]=c[j];c[sb[i]-'a'+1]=i;}
88         memset(da,63,sizeof(da));da[0]=0;ss=0;
89         for(int i=1;i<=len1;i++)
90             for(int j=len2;j>=0;j--)
91             {
92                 ss=last[j][sa[i]-'a'+1];
93                 if(ss>len1)ans=min(ans,da[j]+1);else da[ss]=min(da[ss],da[j]+1);
94             }
95         if(ans==inf)printf("-1\n");
96         else printf("%d\n",ans);
97     //}
98     return 0;
99 }

 

posted @ 2018-04-10 21:34  CHerish_OI  阅读(256)  评论(0编辑  收藏  举报