【题目】

给定一个字符串str,返回str中最长回文子串的长度

【举例】

str="123", 1

str=“abc1234321ab” 7

【暴力破解】

从左到右遍历字符串,遍历到每个字符的时候,以当前字符作为中心能够产生多大的回文字符串,

奇回文和偶 回文寻找方式不一样。

缺点:前面的寻找无法为后面的寻找提供任何帮助。没有记忆。加上记忆就好了。

【Manacher】

Manacher算法解决的问题是在线性时间内找到一个字符串的最长回文子串。

  1. 奇回文和偶回味在判断是比较麻烦,首先对str进行处理。插入 特殊字符。

    123 -》 #1#2#3#

通过这种处理方式,偶回文也有了中心轴。 解决了奇偶的差异性。

特殊字符# 用于对应的是自己。所以对整个求解无影响。

其中用到 i & 1 判断当前索引是 奇数还是偶数。参考这个

https://segmentfault.com/q/1010000021162482/a-1020000021164176

public static char[] manacherString(String str) {
char[] charArr = str.toCharArray();
char[] res = new char[2 * charArr.length + 1];
int index = 0;
for (int i = 0; i < res.length; i++) {
// i&1==0 表示 如果是 偶数
res[i] = (i & 1) == 0 ? '#' : charArr[index++];
}
return res;
}
  1. 处理后的字符串记为charArr.

    需要三个辅助变量
  • pArr: 长度与charArr长度一样。pArr[i]的含义是: 以 i 位置上的字符 charArr[i]作为 回文中心的情况下,扩出去得到的最大回文半径是多少。
  • pR: 这个变量是 之前遍历的所有 字符的所有回文半径中。 最右即将到达的位置。
  • index: 和pR联动。表示最右即将到达的位置 的回文中心位置。

3)从左到右依次算出数组pArr每个位置的值,最大的值 就是处理后的charArr中的最大回文半径。

中括号表示 index的回文左右边界。

小括号表示 i 和 i'的左右回文边界。

分俩大情况, 第二种情况再分为3小情况。

  • 第一种情况

当前的i,在pR之后,就暴力扩就对了。前面的回文信息提供不了任何帮助。

  • 第二种情况

    就是 i在pR之内。

index肯定在 i之前。

因为 i在 某一个元素的回文半径中,所以一定有对称点i'

根据i'的回文半径所处位置划分为三类

1)i'的回文半径 在 index的回文半径内。



i的回文半径就是和i'一样,不可能再大了。

证明:

首先X != Y

Y和Z是 index的对称点。 Y == Z

X和H是 index的对称点。 X == H

所以 Z != H 。所以 i的回文半径长度就是pArr[i']

2) i'的回文半径 超出了index的回文半径



]' [' 代表 以 i' 和 i 为对称的 []的位置。

现在证明 i的 最大回文半径应该是多少呢.

首先 [' 到 ]的位置肯定是回文。只需要看Z和H是否相等。

证明:

X和Y是 以 i'的对称点 。 X == Y。

Y和Z是 以 index的对称点。 Y==Z。 那么 X == Z。

X和H肯定不相等。因为如果相等。那么 index的最大回文半径就不是[]。 X!=H

那么Z!=H

那么 i的回文半径长度 就是 [' 到 ]

3)i'的回文半径 正好 和index的左边界 重合。

那么i最小的回文也是(到]。 因为Z和H是否相无法证明,需要比较了。

以上三种情况,扩出去的过程可以优化,但还是无法 避免扩出去的检查。

public static int maxLcpsLength(String str) {

        if (str == null || str.length() == 0) {
return 0;
} char[] charArr = manacherString(str);
System.out.println(Arrays.toString(charArr));
int[] pArr = new int[charArr.length];
int index = -1;
int pR = -1;
int max = Integer.MIN_VALUE; // 保存最大值。 for (int i = 0; i < charArr.length; i++) { if (pR > i) {
// i 在 pR内,看哪部分不需要比较了。 // i' 的 位置的 回文半径 大小
int pi_ = pArr[2 * index - i]; // 最远也只能扩到 pR . 最小 就是 看i'的回文半径 pi_ int pR_pi_ = pR - i; // i 到 pR的距离长度大小。
// 取最小值.三种情况都符合,也没有多余的浪费。
// 第一种。pR-i 肯定 大于 pi_ 所以没问题
// 第二种. pR-i 肯定 小于 pi_ 本来就是从pR开始扩,也没问题。
// 第三种. pR-i == pi_ 。
pArr[i] = Math.min(pi_, pR_pi_); // 当前 pArr[i] 的值得意思是 以 i为中心,这个半径之内的不需要验证了。后面的还需要比较一下。 } else {
// i 在 pR外。需要自己扩。
pArr[i] = 1;
}
// 以上 整合成一行就是
// pArr[i] = pR > i ? Math.min(pArr[2*index-1], pR-i):1;
while (i + pArr[i] < charArr.length && i - pArr[i] > -1) {
// 扩的停止条件就是 左右边界到了。
if (charArr[i + pArr[i]] == charArr[i - pArr[i]]) {
pArr[i]++;// 左右相等。继续扩。
} else {
// 无法继续扩了,退出。
break;
}
// 更新 pR 和 index ,如果当前 i位置的最右边界 和前面的一样,不更新。只有大于才更新。
if (i + pArr[i] > pR) {
pR = i + pArr[i];
index = i;
}
// 更新max
max = Math.max(max, pArr[i]);
}
} return max - 1; // }

max - 1 ,因为 加了辅助。需要减一。

算法:Manacher,给定一个字符串str,返回str中最长回文子串的长度。的更多相关文章

  1. Manacher算法----最长回文子串

    题目描述 给定一个字符串,求它的最长回文子串的长度. 分析与解法 最容易想到的办法是枚举所有的子串,分别判断其是否为回文.这个思路初看起来是正确的,但却做了很多无用功,如果一个长的子串包含另一个短一些 ...

  2. 最长回文子串—Manacher 算法 及 python实现

    最长回文子串问题:给定一个字符串,求它的最长回文子串长度.如果一个字符串正着读和反着读是一样的,那它就是回文串.   给定一个字符串,求它最长的回文子串长度,例如输入字符串'35534321',它的最 ...

  3. manacher算法求最长回文子串

    一:背景 给定一个字符串,求出其最长回文子串.例如: s="abcd",最长回文长度为 1: s="ababa",最长回文长度为 5: s="abcc ...

  4. leetcode 求一个字符串的最长回文子串

    最长回文子串问题:给定一个字符串,求它的最长回文子串长度.如果一个字符串正着读和反着读是一样的,那它就是回文串.   给定一个字符串,求它最长的回文子串长度,例如输入字符串'35534321',它的最 ...

  5. 51nod1089 最长回文子串 manacher算法

    0. 问题定义 最长回文子串问题:给定一个字符串,求它的最长回文子串长度. 如果一个字符串正着读和反着读是一样的,那它就是回文串.下面是一些回文串的实例: 12321 a aba abba aaaa ...

  6. 最长回文子串——manacher

    最长回文子串--Manacher 算法 (原版的博主的代码都是用py写的,这里改成c++) c++ 算法 字符串处理 0. 问题定义 最长回文子串问题:给定一个字符串,求它的最长回文子串长度. 如果一 ...

  7. hdu3068 求一个字符串中最长回文字符串的长度 Manacher算法

    最长回文 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  8. 最长回文子串(动规,中心扩散法,Manacher算法)

    题目 leetcode:5. Longest Palindromic Substring 解法 动态规划 时间复杂度\(O(n^2)\),空间复杂度\(O(n^2)\) 基本解法直接看代码 class ...

  9. Manacher模板( 线性求最长回文子串 )

    模板 #include<stdio.h> #include<string.h> #include<algorithm> #include<map> us ...

随机推荐

  1. PTA(BasicLevel)-1016 部分A+B

    一.问题定义 正整数 a的"Da(为 1 位整数)部分"定义为由a中所有Da组成的新整数Pa​.例如:给定8,Da = 6,则a的"6 部分"Pa是66,因为a ...

  2. java中的内存划分和一个数组的内存图

    内存概述 内存是计算机中的重要原件,临时存储区域,作用是运行程序.我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存   Java虚拟机要运行程序 ...

  3. 今天安装了eclipse,myeclipse,满满的回忆

    代码半生,编码半世,ideacode失效,安装了eclipse,那熟悉的界面,俨然又回到了从前,当初我们还在用structs,eclipse,webwork,那时候还在用jbuilder,但是算是老套 ...

  4. Hbase学习(三)过滤器 java API

    Hbase学习(三)过滤器 HBase 的基本 API,包括增.删.改.查等. 增.删都是相对简单的操作,与传统的 RDBMS 相比,这里的查询操作略显苍白,只能根据特性的行键进行查询(Get)或者根 ...

  5. 栈和排序_via牛客网

    题目 链接:https://ac.nowcoder.com/acm/contest/26886/A 来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 131072K,其他语 ...

  6. while练习题_1到100之间的偶数和

    依然是while循环四步骤 初始化变量 条件判断 条件执行体 最后就是输出答案就可以了 点击查看笔者代码 a = 1 sum = 0 while a <= 100: if (a+1)%2:#if ...

  7. 造!又有新的生产力语言了「GitHub 热点速览 v.22.30」

    作者:HelloGitHub-小鱼干 你还记得那些 PHP 开发都去哪了吗?转 Golang 了!移动端现在流行什么?Flutter 编程.现在谷歌带着新的生产力语言来了,Carbon,代号:C++ ...

  8. 基于SqlSugar的开发框架循序渐进介绍(13)-- 基于ElementPlus的上传组件进行封装,便于项目使用

    在我们实际项目开发过程中,往往需要根据实际情况,对组件进行封装,以更简便的在界面代码中使用,在实际的前端应用中,适当的组件封装,可以减少很多重复的界面代码,并且能够非常简便的使用,本篇随笔介绍基于El ...

  9. 从零开始搭建Vue2.0项目(一)之快速开始

    从零开始搭建Vue2.0项目(一)之项目快速开始 前言 该样板适用于大型,严肃的项目,并假定您对Webpack和有所了解vue-loader.确保还阅读vue-loader的文档,了解常见的工作流程配 ...

  10. JavaWeb--Cookie与Session

    前言 Java Web 其实就是一个技术的总和,把Web看成一个容器而已主要使用JavaEE技术来实现.在加上各种中间件. 整个javaWeb阶段的内容通过实际的案例贯穿学习, 所涉及到的技术知识点会 ...