51Nod-1277-字符串中的最大值

题目链接:点击打开链接

题目大意:
定义一个字符串的最大值值为:该字符串各个前缀出现的次数与对应前缀的长度的乘积的最大值。给定一个字符串,求该字符串的最大值。
解题思路:
(刚刚才学习了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;
}

说得很迷,准确地说是根本讲不清...,最好自己手算几次,就更容易理解了...。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值