Manacher算法学习 【马拉车】
好久没写算法学习博客了
比较懒,一直在刷水题
今天学一个用于回文串计算问题manacher算法【马拉车】
回文串
回文串:指的是以字符串中心为轴,两边字符关于该轴对称的字符串
——例如abaaba
最大回文子串:一个字符串的最大的子串,满足这个子串是回文串
——例如abcababa的最大回文子串是ababa
求最大回文子串
朴素算法:枚举中心i,向两边扩展,复杂度O(n2)
改进算法:
manacher
朴素算法中,我们在计算以i为中心的回文串时会产生对原先字符的重复遍历,导致效率低下,而manacher通过一些信息的储存来避免了重复遍历
①首先,我们插入一个无关紧要的字符,将所有字符分割开,例如#
——ababa -> #a#b#a#b#a#
这样子以后我们会发现所有回文子串都变成了奇数长度,于是我们可以找到每一个回文子串的中心字符
②算法核心:RL[i]数组,表示以i为中心的最大回文子串的半径长度
——例如#a#b#a#b#a#,我们设最左边一个#是1号,RL[6]=6,RL[8]=4
我们只要算出RL数组就可以求出所有的回文子串了
③算法流程:
从左向右扫,用MR【max right】变量记录当前所有回文子串所能到达的最右位置,pos记录MR所对应的对称中心位置
对于当前位置i,分类讨论:
1、若位于pos和MR之间,我们找到i关于pos对称的位置j,RL[i]至少等于RL[j],且最多延伸到MR
2、若i位于MR右边,RL[i]至少为1,即本身
做完之后,再尝试向两边延伸RL[i],直至不能延伸
更新MR,pos
复杂度分析
我们会发现MR只往右更新,所有在MR左侧的回文子串都可以直接算出,在MR右侧的未知,需要枚举匹配,一旦向右遍历,MR即更新到右侧并覆盖所更新点
说白了就是每个点最多入坑一次
所以复杂度O(n)
板题:POJ2342
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define Redge(u) for (int k = h[u]; k != -1; k = ed[k].nxt)
using namespace std;
const int maxn = 2000005,maxm = 1000005,INF = 1000000000;
inline int RD(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
return out * flag;
}
char T[maxn],s[maxn];
int n,len,RL[maxn];
void manacher(){
int pos = 1,mr = 1; RL[1] = 1;
for (int i = 2; i <= n; i++){
if (i <= mr) RL[i] = min(RL[2 * pos - i],mr - i + 1);
else RL[i] = 1;
while (s[i + RL[i]] == s[i - RL[i]] && i - RL[i]) RL[i]++;
if (i + RL[i] - 1 > mr) mr = i + RL[i] - 1,pos = i;
}
}
int main(){
int cnt = 0;
while (~scanf("%s",T + 1) && T[1] != 'E'){
s[n = 1] = '#';
for (int i = 1; T[i] >= 'a' && T[i] <= 'z'; i++)
s[++n] = T[i],s[++n] = '#';
manacher();
int ans = 0;
REP(i,n) ans = max(ans,RL[i] / 2 * 2 - (s[i] != '#'));
printf("Case %d: %d\n",++cnt,ans);
}
return 0;
}
Manacher算法学习 【马拉车】的更多相关文章
- Manacher算法学习笔记 | LeetCode#5
Manacher算法学习笔记 DECLARATION 引用来源:https://www.cnblogs.com/grandyang/p/4475985.html CONTENT 用途:寻找一个字符串的 ...
- Manacher算法学习笔记
前言 Manacher(也叫马拉车)是一种用于在线性时间内找出字符串中最长回文子串的算法 算法 一般的查找回文串的算法是枚举中心,然后往两侧拓展,看最多拓展出多远.最坏情况下$O(n^2)$ 然而Ma ...
- Manacher算法(马拉车)求最长回文子串
Manacher算法求最长回文字串 算法思路 按照惯例((・◇・)?),这里只是对算法的一些大体思路做一个描述,因为找到了相当好理解的博客可以参考(算法细节见参考文章). 一般而言,我们的判断回文算法 ...
- manacher算法学习(求最长回文子串长度)
Manacher总结 我的代码 学习:yyb luogu题目模板 xzy的模板 #include<iostream> #include<cstdlib> #include< ...
- Manacher算法(马拉车算法)浅谈
什么是Manacher算法? 转载自百度百科 Manachar算法主要是处理字符串中关于回文串的问题的,它可以在 O(n) 的时间处理出以字符串中每一个字符为中心的回文串半径,由于将原字符串处理成两倍 ...
- Manacher 算法学习小记
概要 一个字符串有多少个回文的字串?最多有 \(O(n^2)\) 级别个.但 Manacher 算法却可以用 \(O(n)\) 的时间复杂度解决这个问题.同时 Manacher 算法实现非常简单. 一 ...
- Manacher 算法学习笔记
算法用处: 解决最长回文子串的问题(朴素型). 算法复杂度 我们不妨先看看其他暴力解法的复杂度: \(O(n^3)\) 枚举子串的左右边界,然后再暴力判断是否回文,对答案取 \(max\) . \(O ...
- Manacher算法(马拉车)
学习博客:https://www.cnblogs.com/love-yh/p/7072161.html 首先,得先了解什么是回文串(我之前就不是很了解,汗).回文串就是正反读起来就是一样的,如“abb ...
- 学习笔记 - Manacher算法
Manacher算法 - 学习笔记 是从最近Codeforces的一场比赛了解到这个算法的~ 非常新奇,毕竟是第一次听说 \(O(n)\) 的回文串算法 我在 vjudge 上开了一个[练习],有兴趣 ...
随机推荐
- REST Adapter实现SAP PI中的增强XML/JSON格式转换(转载)
SAP标准的REST adapter有着XML/JSON转换的功能,它很有用,因为一方面SAP PI/PO内部以XML格式处理数据,而另一方面,在处理REST架构风格的时候,JSON才是事实上的格式. ...
- egg的使用方法
1.controller const {ctx,service} = this: let id = ctx.query.id // 获取GET的参数 let body = ctx.request.bo ...
- 获取父窗口元素或者获取iframe中的元素(相同域名下)
jquery方法 在父窗口中获取iframe中的元素 //方法1 $("#iframe的ID").contents().find("iframe中的元素"); ...
- Codeforces Round #460 (Div. 2): D. Substring(DAG+DP+判环)
D. Substring time limit per test 3 seconds memory limit per test 256 megabytes input standard input ...
- C# 设定弹出窗体位置
一.C#中弹出窗口位置 加入命名空间 using System.Drawing using System.Windows.Forms 假定窗口名为form1,则 //窗体位置在屏幕中间 form1.S ...
- 浅谈XX系统跨平台迁移(测试环境)
一 概述 XX系统目前运行在XX-A的云平台上,计划将其迁移至XX-B的云平台. XX系统是java开发,中间组件涉及nginx+keepalived实现各个业务系统之间的高可用,kafka,zook ...
- poj 3685 矩阵问题 查找第K小的值
题意:N阶矩阵Aij= i2 + 100000 × i + j2 – 100000 × j + i × j,求第M小的元素. 思路:双重二分 考虑到,aij是跟着i递增的,所以i可以作为一个二分搜索 ...
- contextmanager 的基本使用
from contextlib import contextmanager 简化 With 语句: class MyResource: def query(self): print ...
- python datetime offset-aware与offset-navie相互转换
python datetime offset-aware与offset-navie相互转换 2016年11月13日 16:20:43 阅读数:2393 有时,我们使用python 的datetime模 ...
- centos使用--centos7.3配置LNMP
目录 1 源的配置 2 安装软件 2.1 安装php7 2.2 安装nginx 2.3 安装mysql 2.4 安装vsftp (ftp登录配置) 3 开机启动设置 4 其它一些配置 4.1 git的 ...