用扩展KMP做简单省力.....
#include <iostream> #include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int maxn=100100; char T[maxn],P[maxn]; int next[maxn],ex[maxn]; void pre_exkmp(char P[]) { int m=strlen(P); next[0]=m; int j=0,k=1; while(j+1<m&&P[j]==P[j+1]) j++; next[1]=j; for(int i=2;i<m;i++) { int p=next[k]+k-1; int L=next[i-k]; if(i+L<p+1) next[i]=L; else { j=max(0,p-i+1); while(i+j<m&&P[i+j]==P[j]) j++; next[i]=j; k=i; } } } void exkmp(char P[],char T[]) { int m=strlen(P),n=strlen(T); pre_exkmp(P); int j=0,k=0; while(j<n&&j<m&&P[j]==T[j]) j++; ex[0]=j; for(int i=1;i<n;i++) { int p=ex[k]+k-1; int L=next[i-k]; if(i+L<p+1) ex[i]=L; else { j=max(0,p-i+1); while(i+j<n&&j<m&&T[i+j]==P[j]) j++; ex[i]=j; k=i; } } } int pos[maxn],sum[maxn],mx=-1; struct ANS { int a,b; }ans[maxn]; int na=0; bool cmp(ANS x,ANS y) { if(x.a!=y.a)return x.a<y.a; return x.b<y.b; } int lisan[maxn],nl; int main() { cin>>P; pre_exkmp(P); int n=strlen(P); for(int i=0;i<n;i++) { pos[next[i]]++; lisan[nl++]=next[i]; mx=max(mx,next[i]); } sort(lisan,lisan+nl); int t=unique(lisan,lisan+nl)-lisan; for(int i=t-1;i>=0;i--) { sum[lisan[i]]=sum[lisan[i+1]]+pos[lisan[i]]; } for(int i=0;i<n;i++) { if(next[i]==n-i) { ans[na++]=(ANS){next[i],sum[next[i]]}; } } sort(ans,ans+na,cmp); printf("%d ",na); for(int i=0;i<na;i++) { printf("%d %d ",ans[i].a,ans[i].b); } return 0; }