★★ 输入文件:string_maxval.in
输出文件:string_maxval.out
简单对比
时间限制:1 s 内存限制:256 MB
【题目描述】
一个字符串的前缀是指包含该字符第一个字母的连续子串,例如:abcd的所有前缀为a, ab, abc, abcd。
给出一个字符串S,求其所有前缀中,字符长度与出现次数的乘积的最大值。
例如:S = "abababa" 所有的前缀如下:
"a", 长度与出现次数的乘积 1 * 4 = 4,
"ab",长度与出现次数的乘积 2 * 3 = 6,
"aba", 长度与出现次数的乘积 3 * 3 = 9,
"abab", 长度与出现次数的乘积 4 * 2 = 8,
"ababa", 长度与出现次数的乘积 5 * 2 = 10,
"ababab", 长度与出现次数的乘积 6 * 1 = 6,
"abababa", 长度与出现次数的乘积 7 * 1 = 7. 其中"ababa"出现了2次,二者的乘积为10,是所有前缀中最大的
【输入格式】
输入字符串T, (1 <= L <= 1000000, L为T的长度),T中的所有字符均为小写英文字母。 (注意:原题是L <= 10W,这里加强一下!)
【输出格式】
输出所有前缀中字符长度与出现次数的乘积的最大值。
【样例输入】
abababa
【样例输出】
10
【解析】
kmp求出每个前缀子串出现的次数*它的长度。
【code】
//cogs 2556 #include<iostream> #include<cstdio> #include<cstring> using namespace std; #define N 1000009 char s[N]; int next[N]; int ans,l,maxx=-1; void init() { scanf("%s",s); l=strlen(s); } void getnext() { next[0]=-1; for(int i=1,j;i<l;i++) { j=next[i-1]; while(s[i]!=s[j+1]&&j>=0) j=next[j]; next[i]=s[i]==s[j+1]?j+1:-1; } } void slove(int ll) { ans=0;//忘记清0; int i=0,j=0; while(i<ll&&j<l) { if(s[i]==s[j]) { i++;j++; } else if(i==0)j++; else i=next[i-1]+1; if(i==ll) { ans++; i=next[i-1]+1; } } if(ans*ll>maxx)maxx=ans*ll;//更新最大值 } int main() { init(); getnext(); for(int i=1;i<=l;i++) slove(i); printf("%d",maxx); return 0; }