回文串(回文树,每种本质不同的字符串的数目*其长度)

考虑一个只包含小写拉丁字母的字符串s。我们定义s的一个子串t的“出 
现值”为t在s中的出现次数乘以t的长度。请你求出s的所有回文子串中的最 
大出现值。 

Input

输入只有一行,为一个只包含小写字母(a -z)的非空字符串s。 

Output


输出一个整数,为逝查回文子串的最大出现值。 

Sample Input

【样例输入l】 abacaba 【样例输入2] www

Sample Output

【样例输出l】 7 【样例输出2] 4

Hint

 



一个串是回文的,当且仅当它从左到右读和从右到左读完全一样。 

在第一个样例中,回文子串有7个:a,b,c,aba,aca,bacab,abacaba,其中: 

● a出现4次,其出现值为4:1:1=4 

● b出现2次,其出现值为2:1:1=2 

● c出现1次,其出现值为l:1:l=l 

● aba出现2次,其出现值为2:1:3=6 

● aca出现1次,其出现值为1=1:3=3 

●bacab出现1次,其出现值为1:1:5=5 

● abacaba出现1次,其出现值为1:1:7=7 

故最大回文子串出现值为7。 

【数据规模与评分】 

数据满足1≤字符串长度≤300000。

 

cnt[i] :i节点代表的回文串的数目

len[i]:i节点代表的回文串的长度

(写回文树有点面向对象的感觉,神器~~~)

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=3e5+10;
char s[N];
struct Palindromic_Tree{
	int next[N][26];//next指针,next指针和字典树类似,指向的串为当前串两端加上同一个字符构成
	int fail[N];//fail指针,失配后跳转到fail指针指向的节点     最长回文后缀 
	int len[N];//len[i]表示节点i表示的回文串的长度  
	int S[N];//存放添加的字符
	ll cnt[N];//结点表示的本质不同的回文串的个数(调用count()后)
	int num[N];//结点表示的最长回文串的最右端点为回文串结尾的回文串个数
	int last;//指向上一个字符所在的节点,方便下一次add 
	int n;//字符数组指针  
    int p;//节点指针
    	
	int newnode(int x){//新建节点 
		memset(next[p],0,sizeof(next[p]));
		cnt[p]=0;
		num[p]=0;
		len[p]=x;
		return p++;
	}
	void init(){
		p=0;
		newnode(0);
		newnode(-1);
		last=0;
		n=0;
		S[0]=-1;//开头放一个字符集中没有的字符,减少特判 
		fail[0]=1;
	}
	int get_fail(int x){//和KMP一样,失配后找一个尽量最长的  
		while(S[n-len[x]-1]!=S[n]) x=fail[x];
		return x;
	} 
	void add(int c){
		c-='a';
		S[++n]=c;
		int cur=get_fail(last);//通过上一个回文串找这个回文串的匹配位置
		if(!next[cur][c]){//如果这个回文串没有出现过,说明出现了一个新的本质不同的回文串  
			int now=newnode(len[cur]+2);//新建节点
			fail[now]=next[get_fail(fail[cur])][c];//和AC自动机一样建立fail指针,以便失配后跳转  
			num[now]=num[fail[now]]+1;
			next[cur][c]=now;
		} 
		last=next[cur][c];
		cnt[last]++;		
	}
	void count(){
		for(int i=p-1;i>=0;i--) cnt[fail[i]]+=cnt[i];
		//父亲累加儿子的cnt,因为如果fail[v]=u,则u一定是v的子回文串!
	}
	ll solve(){
	    ll ans=0;
		for(int i=2;i<p;i++){
			ans=max(ans,cnt[i]*len[i]);
		}
		printf("%lld\n",ans);
	}
}pam;
int main(){
    scanf("%s",s);
    int len=strlen(s);
    pam.init();
    for(int i=0;i<len;i++){
    	pam.add(s[i]);
	}
	pam.count();
	pam.solve();
	return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值