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 上开了一个[练习],有兴趣 ...
随机推荐
- Shell学习——列出当前路径下所有目录
1.ls -d */[root@client02 ~]# ls -d */Desktop/ Documents/ Downloads/ jq-1.5/ Music/ Pictures/ Public/ ...
- Apache Maven(二):构建生命周期
Maven 约定的目录结构 我要遵循Maven已经约定好的目录结构,才能让maven在自动构建过程中找到对应的资源进行构建处理.以下是maven约定的目录结构: 项目名称 |-- pom.xml :M ...
- 初试PHP连接sql server
最开始想使用 pdo_sqlsrv 拓展,但是一直没成功,本文采用的是 pdo_dblib + freetds. 环境:CentOS 6.8.PHP 5.6.20 freetds wget ftp:/ ...
- poj 2674 线性世界 弹性碰撞
弹性碰撞的题目一般都是指碰到就会掉转方向的一类题目,这里我们可以忽略掉头,仅仅看成擦肩而过,交换名字等等 题意:一条线上N只蚂蚁,每只蚂蚁速度固定,方向和坐标不同,碰头后掉头,求最后掉下去那只蚂蚁的名 ...
- 搭建Linpack
环境:vmware workstation14 + centos7(linux基本都可以) 一.开始安装mpich 1. 解决依赖gcc gcc-gfortran sudo yum install g ...
- Android面试收集录6 事件分发机制
转自:秋招面试宝典. 一. 基础认知 1.1 事件分发的对象是谁? 答:事件 当用户触摸屏幕时(View或ViewGroup派生的控件),将产生点击事件(Touch事件). Touch事件相关细节(发 ...
- 剑指Offer - 九度1350 - 二叉树的深度
剑指Offer - 九度1350 - 二叉树的深度2013-11-23 00:54 题目描述: 输入一棵二叉树,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的 ...
- leetcode 【 Remove Duplicates from Sorted List II 】 python 实现
题目: Given a sorted linked list, delete all nodes that have duplicate numbers, leaving only distinct ...
- NOIP 2018 总结
NOIP 2018 总结 提高组: 应得分 \(100 + 100 + 40 + 100 + 50 + 44 = 434\). 考后期望得分 \(100 + 100 + 20 + 100 + 50 + ...
- Centos7中查看IP地址命令ifconfig无法识别如何处理
问题描述: 在虚拟机中已安装好Centos7系统,查看IP地址使用命令ifconfig时,提示找不到此命令,使用ip addr命令则可查询当前系统的IP地址(如图1.2): 图1 图2 解决问题步骤: ...