最长回文子串 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;
}
相关推荐
_李小白22 分钟前
【C++学习笔记】新特性之inline变量
c++·笔记·学习
心中有国也有家24 分钟前
hccl 架构拆解:昇腾集合通信库到底在做什么?
人工智能·经验分享·笔记·分布式·算法·架构
桀人43 分钟前
C++——模板初阶(收录在专栏C++入门到精通)
开发语言·c++
小O的算法实验室1 小时前
2026年MCS,Q-learning增强MOPSO与改进DWA融合算法+复杂三维地形下特定移动机器人动态路径规划
算法
Lumbrologist1 小时前
【C++】零基础入门 · 第 2 节:变量、基本数据类型与输入输出
java·开发语言·c++
XX風1 小时前
CMake / Make / Ninja / MSVC / GCC / Clang / MSBuild —— 完整体系化理解
c++
Peter·Pan爱编程2 小时前
10. new_delete 不是 malloc_free 的包装
c++·人工智能·算法
故事和你913 小时前
洛谷-【动态规划1】动态规划的引入2
开发语言·数据结构·c++·算法·动态规划·图论
重生之我是Java开发战士3 小时前
【动态规划】背包问题:完全背包,二位费用的背包问题,似包非包
算法·动态规划
LabVIEW开发4 小时前
LabVIEW实现FDTD 电磁仿真
算法·labview·labview知识·labview功能·labview程序