【貪心】【後綴自動機】Gym - 101466E - Text Editor

題意:給你兩個串A,B,以及一個整數K,讓你找到B的一個儘量長的前綴,使得其在A串中出現的次數不小於K次。spa

對A串創建後綴自動機,而後把B串放在上面跑,因爲每到一個結點,該結點endpos集合的大小就是該前綴出現的次數,又因爲較長前綴的出現次數不大於較短前綴,因此只要跑到不知足條件停下便可。blog

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAXL 100000
#define MAXC 300
int v[2*MAXL+10],__next[2*MAXL+10],first[2*MAXL+10],e;
void AddEdge(int U,int V){
    v[++e]=V;
    __next[e]=first[U];
    first[U]=e;
}
char s[MAXL+10];
int len;
struct SAM{
    int endcnt[2*MAXL+10];
    int n,maxlen[2*MAXL+10],minlen[2*MAXL+10],trans[2*MAXL+10][MAXC],slink[2*MAXL+10];
    void clear(){
        for(int i=0;i<=n;++i){
            endcnt[i]=maxlen[i]=minlen[i]=slink[i]=first[i]=0;
            memset(trans[i],0,sizeof(trans[i]));
        }
        n=e=0;
    }
    int new_state(int _maxlen,int _minlen,int _trans[],int _slink){
        maxlen[n]=_maxlen;
        minlen[n]=_minlen;
        for(int i=0;i<MAXC;++i){
            if(_trans==NULL){
                trans[n][i]=-1;
            }
            else{
                trans[n][i]=_trans[i];
            }
        }
        slink[n]=_slink;
        return n++;
    }
    int add_char(char ch,int u){
        if(u==-1){
            return new_state(0,0,NULL,-1);
        }
        int c=ch;
        int z=new_state(maxlen[u]+1,-1,NULL,-1);
        endcnt[z]=1;
        int v=u;
        while(v!=-1 && trans[v][c]==-1){
            trans[v][c]=z;
            v=slink[v];
        }
        if(v==-1){
            minlen[z]=1;
            slink[z]=0;
            return z;
        }
        int x=trans[v][c];
        if(maxlen[v]+1==maxlen[x]){
            minlen[z]=maxlen[x]+1;
            slink[z]=x;
            return z;
        }
        int y=new_state(maxlen[v]+1,-1,trans[x],slink[x]);
        slink[y]=slink[x];
        minlen[x]=maxlen[y]+1;
        slink[x]=y;
        minlen[z]=maxlen[y]+1;
        slink[z]=y;
        int w=v;
        while(w!=-1 && trans[w][c]==x){
            trans[w][c]=y;
            w=slink[w];
        }
        minlen[y]=maxlen[slink[y]]+1;
        return z;
    }
    void dfs(int U){
        for(int i=first[U];i;i=__next[i]){
            dfs(v[i]);
            endcnt[U]+=endcnt[v[i]];
        }
    }
    void work_slink_tree(){
        for(int i=1;i<n;++i){
            AddEdge(slink[i],i);
        }
        dfs(0);
    }
}sam;
typedef long long ll;
char s2[MAXL+10];
int len2,K;
int main(){
	gets(s);
    len=strlen(s);
    int U=sam.add_char(0,-1);
    for(int i=0;i<len;++i){
        U=sam.add_char(s[i],U);
    }
    sam.work_slink_tree();
    gets(s2);
    len2=strlen(s2);
    scanf("%d",&K);
    int ans=0;
    U=0;
    for(int i=0;i<len2;++i){
    	if(sam.endcnt[U]<K){
    		ans=i-1;
    		goto OUT;
    	}
    	else if(sam.trans[U][s2[i]]==-1){
    		ans=i;
    		goto OUT;
    	}
    	U=sam.trans[U][s2[i]];
    }
    ans=len2;
    if(sam.endcnt[U]<K){
    	ans=len2-1;
    	goto OUT;
    }
    OUT:
    if(ans==0){
    	puts("IMPOSSIBLE");
    }
    else{
    	for(int i=0;i<ans;++i){
    		putchar(s2[i]);
    	}
    	puts("");
    }
    return 0;
}
相關文章
相關標籤/搜索