好久没写算法学习博客了
比较懒,一直在刷水题
今天学一个用于回文串计算问题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

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<algorithm>
  5. #define LL long long int
  6. #define REP(i,n) for (int i = 1; i <= (n); i++)
  7. #define Redge(u) for (int k = h[u]; k != -1; k = ed[k].nxt)
  8. using namespace std;
  9. const int maxn = 2000005,maxm = 1000005,INF = 1000000000;
  10. inline int RD(){
  11. int out = 0,flag = 1; char c = getchar();
  12. while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
  13. while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
  14. return out * flag;
  15. }
  16. char T[maxn],s[maxn];
  17. int n,len,RL[maxn];
  18. void manacher(){
  19. int pos = 1,mr = 1; RL[1] = 1;
  20. for (int i = 2; i <= n; i++){
  21. if (i <= mr) RL[i] = min(RL[2 * pos - i],mr - i + 1);
  22. else RL[i] = 1;
  23. while (s[i + RL[i]] == s[i - RL[i]] && i - RL[i]) RL[i]++;
  24. if (i + RL[i] - 1 > mr) mr = i + RL[i] - 1,pos = i;
  25. }
  26. }
  27. int main(){
  28. int cnt = 0;
  29. while (~scanf("%s",T + 1) && T[1] != 'E'){
  30. s[n = 1] = '#';
  31. for (int i = 1; T[i] >= 'a' && T[i] <= 'z'; i++)
  32. s[++n] = T[i],s[++n] = '#';
  33. manacher();
  34. int ans = 0;
  35. REP(i,n) ans = max(ans,RL[i] / 2 * 2 - (s[i] != '#'));
  36. printf("Case %d: %d\n",++cnt,ans);
  37. }
  38. return 0;
  39. }

Manacher算法学习 【马拉车】的更多相关文章

  1. Manacher算法学习笔记 | LeetCode#5

    Manacher算法学习笔记 DECLARATION 引用来源:https://www.cnblogs.com/grandyang/p/4475985.html CONTENT 用途:寻找一个字符串的 ...

  2. Manacher算法学习笔记

    前言 Manacher(也叫马拉车)是一种用于在线性时间内找出字符串中最长回文子串的算法 算法 一般的查找回文串的算法是枚举中心,然后往两侧拓展,看最多拓展出多远.最坏情况下$O(n^2)$ 然而Ma ...

  3. Manacher算法(马拉车)求最长回文子串

    Manacher算法求最长回文字串 算法思路 按照惯例((・◇・)?),这里只是对算法的一些大体思路做一个描述,因为找到了相当好理解的博客可以参考(算法细节见参考文章). 一般而言,我们的判断回文算法 ...

  4. manacher算法学习(求最长回文子串长度)

    Manacher总结 我的代码 学习:yyb luogu题目模板 xzy的模板 #include<iostream> #include<cstdlib> #include< ...

  5. Manacher算法(马拉车算法)浅谈

    什么是Manacher算法? 转载自百度百科 Manachar算法主要是处理字符串中关于回文串的问题的,它可以在 O(n) 的时间处理出以字符串中每一个字符为中心的回文串半径,由于将原字符串处理成两倍 ...

  6. Manacher 算法学习小记

    概要 一个字符串有多少个回文的字串?最多有 \(O(n^2)\) 级别个.但 Manacher 算法却可以用 \(O(n)\) 的时间复杂度解决这个问题.同时 Manacher 算法实现非常简单. 一 ...

  7. Manacher 算法学习笔记

    算法用处: 解决最长回文子串的问题(朴素型). 算法复杂度 我们不妨先看看其他暴力解法的复杂度: \(O(n^3)\) 枚举子串的左右边界,然后再暴力判断是否回文,对答案取 \(max\) . \(O ...

  8. Manacher算法(马拉车)

    学习博客:https://www.cnblogs.com/love-yh/p/7072161.html 首先,得先了解什么是回文串(我之前就不是很了解,汗).回文串就是正反读起来就是一样的,如“abb ...

  9. 学习笔记 - Manacher算法

    Manacher算法 - 学习笔记 是从最近Codeforces的一场比赛了解到这个算法的~ 非常新奇,毕竟是第一次听说 \(O(n)\) 的回文串算法 我在 vjudge 上开了一个[练习],有兴趣 ...

随机推荐

  1. REST Adapter实现SAP PI中的增强XML/JSON格式转换(转载)

    SAP标准的REST adapter有着XML/JSON转换的功能,它很有用,因为一方面SAP PI/PO内部以XML格式处理数据,而另一方面,在处理REST架构风格的时候,JSON才是事实上的格式. ...

  2. egg的使用方法

    1.controller const {ctx,service} = this: let id = ctx.query.id // 获取GET的参数 let body = ctx.request.bo ...

  3. 获取父窗口元素或者获取iframe中的元素(相同域名下)

    jquery方法 在父窗口中获取iframe中的元素 //方法1 $("#iframe的ID").contents().find("iframe中的元素"); ...

  4. 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 ...

  5. C# 设定弹出窗体位置

    一.C#中弹出窗口位置 加入命名空间 using System.Drawing using System.Windows.Forms 假定窗口名为form1,则 //窗体位置在屏幕中间 form1.S ...

  6. 浅谈XX系统跨平台迁移(测试环境)

    一 概述 XX系统目前运行在XX-A的云平台上,计划将其迁移至XX-B的云平台. XX系统是java开发,中间组件涉及nginx+keepalived实现各个业务系统之间的高可用,kafka,zook ...

  7. poj 3685 矩阵问题 查找第K小的值

    题意:N阶矩阵Aij= i2 + 100000 × i + j2 – 100000 × j + i × j,求第M小的元素. 思路:双重二分 考虑到,aij是跟着i递增的,所以i可以作为一个二分搜索 ...

  8. contextmanager 的基本使用

    from contextlib import contextmanager 简化 With 语句: class MyResource:    def query(self):        print ...

  9. python datetime offset-aware与offset-navie相互转换

    python datetime offset-aware与offset-navie相互转换 2016年11月13日 16:20:43 阅读数:2393 有时,我们使用python 的datetime模 ...

  10. centos使用--centos7.3配置LNMP

    目录 1 源的配置 2 安装软件 2.1 安装php7 2.2 安装nginx 2.3 安装mysql 2.4 安装vsftp (ftp登录配置) 3 开机启动设置 4 其它一些配置 4.1 git的 ...