其实就像题面描述的那样……主要还是KMP算法的基础上搞一些DP(递推)之类……
随便yy一下就可以得到50分了,因为num数组跟next的关系还是很密切的……
f[i]=f[next[i]]+1;
然后取 t=next[i]; while(t*2>i) t=next[t]; num[i]=num[t]+1;
然而我这样写……TLE了QwQ
因为这样每次都需要重新定位t……
所以为了省时!我们直接用一个全局指针来搞……(UOJ是个宝地压~参考了RXDoi 神犇的代码)
P.S.这次自己重新yy KMP算法还是有些收获的,不过好像让字符串从0开始和从1开始,KMP会有些不同……?然而我一开始写的从0开始,怎么也yy不出全局指针的写法……sad
1 /************************************************************** 2 Problem: 3670 3 User: Tunix 4 Language: C++ 5 Result: Accepted 6 Time:628 ms 7 Memory:10060 kb 8 ****************************************************************/ 9 10 //BZOJ 3670 11 #include<vector> 12 #include<cstdio> 13 #include<cstring> 14 #include<cstdlib> 15 #include<iostream> 16 #include<algorithm> 17 #define rep(i,n) for(int i=0;i<n;++i) 18 #define F(i,j,n) for(int i=j;i<=n;++i) 19 #define D(i,j,n) for(int i=j;i>=n;--i) 20 #define pb push_back 21 using namespace std; 22 inline int getint(){ 23 int v=0,sign=1; char ch=getchar(); 24 while(ch<'0'||ch>'9'){ if (ch=='-') sign=-1; ch=getchar();} 25 while(ch>='0'&&ch<='9'){ v=v*10+ch-'0'; ch=getchar();} 26 return v*sign; 27 } 28 const int N=1e6+10,INF=~0u>>2,P=1e9+7; 29 typedef long long LL; 30 /******************tamplate*********************/ 31 32 //#define debug 33 int n,nxt[N],f[N]; 34 char s[N]; 35 36 int main(){ 37 #ifndef ONLINE_JUDGE 38 freopen("3670.in","r",stdin); 39 freopen("3670.out","w",stdout); 40 #endif 41 int T=getint(); 42 while(T--){ 43 scanf("%s",s+1); 44 n=strlen(s+1); 45 nxt[1]=0; f[1]=1; 46 int ans=1; 47 int j=0,k=0; 48 F(i,2,n){ 49 while(j && s[i]!=s[j+1]) j=nxt[j]; 50 while(k && s[i]!=s[k+1]) k=nxt[k]; 51 if (s[i]==s[j+1]) j++; 52 if (s[i]==s[k+1]) k++; 53 nxt[i]=j; f[i]=f[j]+1; 54 while(k*2>i) k=nxt[k]; 55 ans=(LL)ans*(f[k]+1)%P; 56 } 57 #ifdef debug 58 printf("%s\n",s); 59 printf("nxt : "); 60 rep(i,n) printf("%d ",nxt[i]); puts(""); 61 printf("f : "); 62 rep(i,n) printf("%d ",f[i]); puts(""); 63 #endif 64 printf("%d\n",ans); 65 } 66 return 0; 67 }