题目链接:点击打开链接
题目大意:
定义一个字符串的最大值值为:该字符串各个前缀出现的次数与对应前缀的长度的乘积的最大值。给定一个字符串,求该字符串的最大值。
解题思路:
(刚刚才学习了KMP()算法,用地还不是很熟练。)
对于Next[]数组,Next[i]==j可以理解为在第i的字符前面有j个字符与长度为j的前缀相同,那么当我们求出给定字符串的Next[]数组后,就知道了对应字符串的每一个字符前面最长的与某前缀相同的子串的长度,比如Next[i]==j就表示在位置i的前面j个字符与长度为j的前缀相同。那么我们求长度为j的前缀出现的个数时,就把所有满足Next[i]==j的长度为i的前缀的出现次数加起来就行了。j相同,不同的i对应的子串也就不同,说明不存在重复问题。
代码:
#include<iostream>
#include<cstring>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define LL long long
using namespace std;
const int maxn=1e5+10;
string str;
int vis[maxn];
int len,Next[maxn];
void GetNext(){
int i=0,j=-1;
Next[0]=-1;
len=str.length();
while(i<len){
if(j==-1||str[i]==str[j])Next[++i]=++j;
else j=Next[j];
}
}
int main(){
freopen("in.txt","r",stdin);
cin.tie(0);
ios_base::sync_with_stdio(0);
cin>>str;
GetNext();
for(int i=0;i<=len;i++)cout<<Next[i]<<" ";cout<<endl;
for(int i=1;i<=len;i++)vis[i]=1;
LL mx=-1;
for(int i=len;i>=1;i--){vis[Next[i]]+=vis[i];}
for(int i=0;i<=len;i++){
if(mx<(LL)vis[i]*i)mx=(LL)vis[i]*i;
}
cout<<mx<<endl;
return 0;
}
说得很迷,准确地说是根本讲不清...,最好自己手算几次,就更容易理解了...。