链接:
http://acm.hdu.edu.cn/showproblem.php?pid=4333
题目大意:
给一个数字字符串S, 可以把S最后一个数字移动到最前面变成另一个数字。例如123, 经过移动依次变成312,231,123。 注意当移动次数正好和S长度相等时,S又变回了最开始的那个数字。
求这个移动过程所形成的所有字符串,大于S(最初的)的数字,等于S,以及小于S的各有多少个。
分析与总结:
1. 首先要考虑以怎样的形式保存S,因为以往题目一般的移动方式都是最左边的数字移动到最右边的,只需要复制2倍的即可。 而这题是把最右边的移动到最左边的,也可以这样处理。虽然它是最右边的移到最左边的,但是我们可以逆过来想,从最终状态变为就初状态,就相当于“恢复”的过程:把最左边的数字移动到最右边。
2. 然后是枚举起点比较。如果朴素的方法复杂度为O(n^2),肯定超时,所以要想到用一个线性时间的方法。
再仔细一看,发现其实就是裸的拓展KMP了:设原来字符串为T, 长度为len, 那么复制两倍后的为S, S的前len个数就是最初的数字。接下来求T的所有后缀与T的最长公共长度。求这个,是为了节省比较的时间,假设有两个字符串aabbde, aaberf,那么已经知道了前面三个数字是相同的了,只需要比较第4个数即可。 这样,比较只需要O(1)的复杂度即可。那么总的复杂度就是O(n).
3. 这题还要考虑去重的情况,可能移动的过程会有多个相同的数字。之所以会有多个相同的数,是因为有循环节。所以只需要求出最小循环节,枚举这最小循环节之内的情况即可。
4. 拓展KMP求最短循环节的方法:
// 已经求出next数组 int kk; // kk保存最短循环节 for(int i=1; i<=len; ++i){ if(i+next[i]>=len){ kk = len%i ? len : i; break; } }
代码:
#include<iostream> #include<cstdio> #include<cstring> using namespace std; const int MAXN = 10^100000*2; char S[MAXN]; int f[MAXN]; void getNext(char* T, int* next){ int len=strlen(T), a=0; next[0] = len; while(a<len-1 && T[a]==T[a+1]) ++a; next[1] = a; a=1; for(int k=2; k<len; ++k){ int p=a+next[a]-1, L=next[k-a]; if(k-1+L >= p){ int j = max(p-k+1, 0); while(k+j<len && T[k+j]==T[j]) ++j; next[k] = j; a=k; } else next[k] = L; } } int main(){ int cas=1,nCase; scanf("%d",&nCase); while(nCase--){ scanf("%s",S); int len=strlen(S); for(int i=0; i<len; ++i){ S[i+len] = S[i]; } S[2*len] = '\0'; getNext(S,f); int L=0, E=0, G=0; // 拓展KMP求最短循环节 int kk; for(int i=1; i<=len; ++i){ if(i+f[i]>=len){ kk = len%i ? len : i; break; } } for(int i=0; i<kk; ++i){ if(f[i]>=len) ++E; else if(f[i]>0 && f[i]<len){ if(S[i+f[i]] > S[f[i]]) ++G; else ++L; } else if(f[i]==0){ if(S[i]>S[0]) ++G; else ++L; } } printf("Case %d: %d %d %d\n",cas++,L,E>0,G); } return 0; }
—— 生命的意义,在于赋予它意义士。
原创 http://blog.csdn.net/shuangde800 , By D_Double (转载请标明)