cogs 2566 字符串最大值

2566. [51nod 1129] 字符串最大值

★★   输入文件: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

——————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————
题意:找字符串的各个前缀在字符串中出现的次数,从而的到最大值。

开始认为就是求next[]然后用next[]求各个前缀在字符串中出现的次数,结果只有60分。后来看了其他同学的代码发现了这个比较巧妙的方法。即:任意前缀出现的次数,这个前缀的next(前缀的公共前后缀中的前缀)也会回出现这些次数。
——————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————
 1 #include<cstdio>
 2 #include<iostream>
 3 #include<cstring>
 4 
 5 using namespace std;
 6 const int maxl=1000010;
 7 char s[maxl],su[maxl];
 8 int next[maxl],l,ans=0,cs[maxl];
 9 void getnext()
10 {
11     next[0]=-1;
12     l=strlen(s);
13     for(int j,i=1;i<l;i++)
14     {
15         j=next[i-1];
16         while(s[i]!=s[j+1]&&j>=0)j=next[j];
17         next[i]=s[i]==s[j+1]?j+1:-1;
18     }
19 }
20 int main()
21 {
22     freopen("string_maxval.in","r",stdin);
23     freopen("string_maxval.out","w",stdout);
24     scanf("%s",s);
25     getnext();
26     for(int i=l-1;i>=0;--i)
27     {
28         cs[i]++;
29         cs[next[i]]+=cs[i];
30         if(cs[i]*(i+1)>ans)ans=cs[i]*(i+1);
31     }
32     cout<<ans;
33     fclose(stdin);fclose(stdout);
34     return 0;
35 }
View Code


原文地址:https://www.cnblogs.com/gryzy/p/6523905.html