最长回文子串 V2(Manacher算法)

题目描述

回文串是指aba、abba、cccbccc、aaaa这种左右对称的字符串。

输入一个字符串Str,输出Str里最长回文子串的长度。

输入格式

输入Str(Str的长度 <= 100000)

输出格式

输出最长回文子串的长度L。

样例

【样例输入】

复制代码
daabaac

【样例输出】

复制代码
5

一些想法

这道题用 Manacher 算法做会更高效且简单。

算法的优化方法主要是可以不用多次求新的回文子串长度,可以直接用已有答案求出范围内答案。

定义变量:int 类型的有原字符串长度、新字符串长度和回文子串半径长度数组。char 类型的有 原字符串和新字符串(都是数组)。

主函数:输入原字符串加一,获取原字符串长度加一(要加一是要将从下标 0 开始移到从下标 1 开始,方便添加"哨兵"),然后直接调用 Manacher 函数输出。

Manacher 自定义函数:新字符串长度等于原字符串长度 乘 2 加 1,乘二是因为每个数前面要加一个字符,加一是因为要在新字符串最后也加一个"哨兵"字符。然后添加哨兵字符,新字符串第 0 位 和最后一位(m+1)等于一个不同的字符,以防越界(避免后面判断越界,更简便),然后将新字符串第一位初始化为分隔字符。循环原字符串每一个字符,将新字符串中奇数下标的字符等于分隔符,偶数下标的字符等于对应的原字符串的字符。(这里是在构建新字符串)

将用于记录最右回文子串中心点和最右点的变量初始化。循环新字符串每一个字符,初始每个字符的回文半径长度等于 1(自己也是一个回文子串,长度为一)。如果当前数在最右回文子串范围内,说明可以有根据求当前回文半径。将当前回文半径取当前点的对称点的回文半径和最右点到当前点的距离的最小值(因为对称点有可能会超出当前最右回文子串的范围,无法保证超出部分回文,如果超出,直接用最右点到当前点的距离,将超出部分暂时抛开)。

扩展半径。如果在当前点在新字符串中超出的左部分和超出的右部分相等,说明超出部分也回文,可以扩展半径,将当前回文半径加一。

更新最右回文子串的中心点和最右点。如果新的回文子串的最右点比当前记录的最右点大,说明需要更新,更新两个点。(方便后面的数计算)

找完所有点(中心点)的回文半径后,找当中的最大值。循环整个新字符串,找到最大的回文半径减一(减一原因看上一篇或看下面)。返回最大值。

这里复习一遍 Manacher 算法的描述与证明:

AC代码

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int n,m,p[1000005];
char ch[1000005],st[1000005];
int manacher(){
	m=2*n+1;
	st[0]='+',st[m+1]='-',st[1]='#';
	for(int i=1;i<=n;i++){
		st[i<<1]=ch[i];
		st[i<<1|1]='#';
	}
	int maxid=0,id;
	for(int i=1;i<=m;i++){
		p[i]=1;
		if(i<maxid){
			p[i]=min(maxid-i,p[id*2-i]);
		}
		while(st[p[i]+i]==st[i-p[i]]){
			p[i]++;
		}
		if(p[i]+i>maxid){
			maxid=p[i]+i;
			id=i; 
		}
	}
	int maxn=0;
	for(int i=1;i<=m;i++){
		maxn=max(maxn,p[i]-1);
	}
	return maxn;
}
int main(){
	scanf("%s",ch+1);
	n=strlen(ch+1);
	cout<<manacher();
	return 0;
}
相关推荐
Evand J2 小时前
【MATLAB复现RRT(快速随机树)算法】用于二维平面上的无人车路径规划与避障,含性能分析与可视化
算法·matlab·平面·无人车·rrt·避障
一招定胜负2 小时前
机器学习+深度学习经典算法面试复习指南
深度学习·算法·机器学习
皮卡狮2 小时前
高阶数据结构:AVL树
数据结构·算法
天若有情6732 小时前
通用个性化推荐核心架构思路:从视频到电商的跨场景落地实践
人工智能·算法·架构·推流·个性化推荐·猜你喜欢
s09071362 小时前
【声纳成像】基于滑动子孔径与加权拼接的条带式多子阵SAS连续成像(MATLAB仿真)
开发语言·算法·matlab·合成孔径声呐·后向投影算法·条带拼接
jay神2 小时前
基于YOLOv8的钢材表面缺陷检测系统
人工智能·算法·yolo·目标检测·计算机视觉
不想看见4042 小时前
C++八股文【详细总结】
java·开发语言·c++
Accerlator2 小时前
2026年3月21日刷题
算法
2401_891655812 小时前
此电脑网络位置异常的AD域排错指南的技术文章大纲
开发语言·python·算法