最长回文子串 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;
}
相关推荐
leoufung15 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了15 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL15 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰15 小时前
C++ 排列组合完整指南
开发语言·c++·算法
代码中介商15 小时前
银行管理系统的业务血肉 —— 流程、状态机、输入校验与持久化(下篇)
c语言·算法
橙子也要努力变强16 小时前
信号捕捉底层机制-机理篇2
linux·服务器·c++
foundbug99916 小时前
自适应滤除直达波干扰的MATLAB实现
开发语言·算法·matlab
盐焗鹌鹑蛋16 小时前
【C++】stack和queue类
c++
郝学胜-神的一滴17 小时前
罗德里格斯旋转公式(Rodrigues‘ Rotation Formula)完整推导
c++·unity·godot·图形渲染·three.js·unreal
lzh2004091917 小时前
深入理解进程:从PCB内核结构到写时拷贝的底层实战
linux·c++