转载自  最长回文子串(Longest Palindromic Substring)——三种时间复杂度的解法

子串:小于等于原字符串长度由原字符串中任意个连续字符组成的子序列

回文:关于中间字符对称的文法,即“aba”(单核)、“cabbac”(双核)等

最长回文子串:1.寻找回文子串;2.该子串是回文子串中长度最长的。

一、O(n^3)时间复杂度方法——暴力求解

1.思想:

1)从最长的子串开始,遍历所有该原字符串的子串;

2)每找出一个字符串,就判断该字符串是否为回文;

3)子串为回文时,则找到了最长的回文子串,因此结束;反之,则继续遍历。

2.时间复杂度解释:

遍历字符串子串:嵌套一个循环、O(n^2);

判断是否为回文:再次嵌套一个循环、O(n^3)。

3.java代码详解:

   public static String longestPalindrome(String s) {if(s.length() <= 1)return s;for(int i = s.length();i > 0; i--) {//子串长度for (int j = 0; j <= s.length() - i; j++) {String sub = s.substring(j , i + j);//子串位置int count = 0;//计数,用来判断是否对称for (int k = 0; k < sub.length() / 2; k++) {//左右对称判断if (sub.charAt(k) == sub.charAt(sub.length() - k - 1))count++;}if (count == sub.length() / 2)return sub;}}return "";//表示字符串中无回文子串}

二、O(n^2)时间复杂度方法——从中心向外扩散

1.思想:

1)将子串分为单核和双核的情况,单核即指子串长度为奇数,双核则为偶数;

2)遍历每个除最后一个位置的字符index(字符位置),单核:初始low = 初始high = index,low和high均不超过原字符串的下限和上限;判断low和high处的字符是否相等,相等则low++、high++(双核:初始high = 初始low+1 = index + 1);

3)每次low与high处的字符相等时,都将当前最长的回文子串长度与high-low+1比较。后者大时,将最长的回文子串改为low与high之间的;

4)重复执行2)、3),直至high-low+1 等于原字符串长度或者遍历到最后一个字符,取当前截取到的回文子串,该子串即为最长的回文子串。

2.时间复杂度解释:

遍历字符:一层循环、O(n-1);

找以当前字符为中心的最长回文子串:嵌套两个独立循环、O(2n*(n-1)) = O(n^2)。

3.java代码详解:

private static int maxLen = 0;private static String sub = "";public static String longestPalindrome(String s) {if(s.length() <= 1)return s;for(int i = 0;i < s.length()-1;i++){findLongestPalindrome(s,i,i);//单核回文findLongestPalindrome(s,i,i+1);//双核回文}return sub;}public static  void findLongestPalindrome(String s,int low,int high){while (low >= 0 && high <= s.length()-1){if(s.charAt(low) == s.charAt(high)){if(high - low + 1 > maxLen){maxLen = high - low + 1;sub = s.substring(low , high+1);}low --;//向两边扩散找当前字符为中心的最大回文子串high ++;}elsebreak;}}

三、O(n)时间复杂度方法——Manacher算法

1.思想:

1)将原字符串S的每个字符间都插入一个永远不会在S中出现的字符(本例中用“#”表示),在S的首尾也插入该字符,使得到的新字符串S_new长度为2*S.length()+1,保证Len的长度为奇数(下例中空格不表示字符,仅美观作用);

例:S:       a  a  b  a  b  b  a

S_new:        #  a  #  a  #  b  #  a  #  b  #  b  #  a  #

2)根据S_new求出以每个字符为中心的最长回文子串的最右端字符距离该字符的距离,存入Len数组中,(i为回文中间位置,r为回文右端位置)即S_new[i]—S_new[r]为S_new[i]的最长回文子串的右段(S_new[2i-r]—S_new[r]为以S_new[i]为中心的最长回文子串),Len[i] = r - i + 1;

S_new:        #  a  #  a  #  b  #  a  #  b  #  b  #  a  #

Len:            1  2  3  2  1   4  1  4  1  2  5   2  1  2  1

Len数组性质:Len[i] - 1即为以Len[i]为中心的最长回文子串在S中的长度。在S_new中,以S_new[i]为中心的最长回文子串长度为2Len[i] - 1,由于在S_new中是在每个字符两侧都有新字符“#”,观察可知“#”的数量一定是比原字符多1的,即有Len[i]个,因此真实的回文子串长度为Len[i] - 1,最长回文子串长度为Math.max(Len) - 1。

3)Len数组求解(线性复杂度(O(n))):

a.遍历S_new数组,i为当前遍历到的位置,即求解以S_new[i]为中心的最长回文子串的Len[i];

b.设置两个参数:sub_midd = Len.indexOf(Math.max(Len)表示在i之前所得到的Len数组中的最大值所在位置、sub_side = sub_midd + Len[sub_midd] - 1表示以sub_midd为中心的最长回文子串的最右端在S_new中的位置(原文作者算法思想错误,sub_midd是最右回文的右边界)。起始sub_midd和sub_side设为0,从S_new中的第一个字母开始计算,每次计算后都需要更新sub_midd和sub_side;

c.当i < sub_side时,取i关于sub_midd的对称点j(j = 2sub_midd - i,由于i <= sub_side,因此2sub_midd - sub_side <= j <= sub_midd);当Len[j] < sub_side - i时,即以S_new[j]为中心的最长回文子串是在以S_new[sub_midd]为中心的最长回文子串的内部,再由于i、j关于sub_midd对称,可知Len[i] = Len[j];

当Len[j] >= sub.side - i时说明以S_new[i]为中心的回文串可能延伸到sub_side之外,而大于sub_side的部分还没有进行匹配,所以要从sub_side+1位置开始进行匹配,直到匹配失败以后,从而更新sub_side和对应的sub_midd以及Len[i];


        d.当i > sub_side时,则说明以S_new[i]为中心的最长回文子串还没开始匹配寻找,因此需要一个一个进行匹配寻找,结束后更新sub_side和对应的sub_midd以及Len[i]。

2.时间复杂度解释:

算法只有遇到还没匹配的位置时才进行匹配,已经匹配过的位置不再进行匹配,因此大大的减少了重复匹配的步骤,对于S_new中的每个字符只进行一次匹配。所以该算法的时间复杂度为O(2n+1)—>O(n)(n为原字符串的长度),所以其时间复杂度依旧是线性的。

3.java代码详解:

public String longestPalindrome(String s) {List<Character> s_new = new ArrayList<>();for(int i = 0;i < s.length();i++){s_new.add('#');s_new.add(s.charAt(i));}s_new.add('#');List<Integer> Len = new ArrayList<>();String sub = "";//最长回文子串int sub_midd = 0;//表示在i之前所得到的Len数组中的最大值所在位置int sub_side = 0;//表示以sub_midd为中心的最长回文子串的最右端在S_new中的位置Len.add(1);for(int i = 1;i < s_new.size();i++){if(i < sub_side) {//i < sub_side时,在Len[j]和sub_side - i中取最小值,省去了j的判断int j = 2 * sub_midd - i;if(j >= 2 * sub_midd - sub_side &&  Len.get(j) <= sub_side - i){Len.add(Len.get(j));}elseLen.add(sub_side - i + 1);}else//i >= sub_side时,从头开始匹配Len.add(1);while( (i - Len.get(i) >= 0 && i + Len.get(i) < s_new.size()) && (s_new.get(i - Len.get(i)) == s_new.get(i + Len.get(i))))Len.set(i,Len.get(i) + 1);//s_new[i]两端开始扩展匹配,直到匹配失败时停止//原文作者算法思想错误,这里应该是Len.get(i) + i - 1 >= Len.get(sub_midd) + sub_midd - 1,求最右端回文边界if(Len.get(i) >= Len.get(sub_midd) ){//匹配的新回文子串长度大于原有的长度sub_side = Len.get(i) + i - 1;sub_midd = i;}}sub = s.substring((2*sub_midd - sub_side)/2,sub_side /2);//在s中找到最长回文子串的位置return sub;}

最长回文子串(Longest Palindromic Substring)相关推荐

  1. [译+改]最长回文子串(Longest Palindromic Substring) Part II

    [译+改]最长回文子串(Longest Palindromic Substring) Part II 原文链接在http://leetcode.com/2011/11/longest-palindro ...

  2. Leetcode 5. 最长回文子串(Longest Palindromic Substring)

    推荐理由:暴力解法太 naive,中心扩散不普适,Manacher 就更不普适了,是专门解这个问题的方法.而用动态规划我认为是最有用的,可以帮助你举一反三的方法. 补充说明:Manacher 算法有兴 ...

  3. leetcode题库:5.最长回文子串Longest Palindrome string

    题目描述: /** 题目地址:https://leetcode-cn.com/problems/longest-palindromic-substring/description/  * 题目:最长回 ...

  4. 最长回文子串与最长回文子序列

    文章目录 最长回文子串与最长回文子序列 最长回文子串 题目描述 dp解法 从中心扩展 马拉车算法 最长回文子序列 题目描述 dp 解法 递归思想 最长回文子串与最长回文子序列 最长回文子串 LeetC ...

  5. leetcode 5 :Longest Palindromic Substring 找出最长回文子串

    题目: Given a string S, find the longest palindromic substring in S. You may assume that the maximum l ...

  6. LeetCode:Longest Palindromic Substring 最长回文子串

    Given a string S, find the longest palindromic substring in S. You may assume that the maximum lengt ...

  7. 21.Longest Palindromic Substring(最长回文子串)

    Level:   Medium 题目描述: Given a string s, find the longest palindromic substring in s. You may assume ...

  8. leetcode(5)—— Longest Palindromic Substring(最长回文子串)

    longest-palindromic-substring 法1:暴力搜索(但会超时) 遍历全部子串(n+(n−1)+-+1=n(1+n)2n+(n-1)+\ldots+1=\frac{n(1+n)} ...

  9. LeetCode 5. Longest Palindromic Substring 最长回文子串 Python 四种解法(Manacher 动态规划)

    Longest Palindromic Substring 最长回文子串 学习笔记 1. Brute method 第一种方法:直接循环求解,o(n2)o(n^2) class Solution:de ...

最新文章

  1. 不对全文内容进行索引的 Loki 到底优秀在哪里
  2. GLSL实现图像处理
  3. 题目1181:遍历链表
  4. 解码resources时里面是空的_深度解码超实用的OpenStack Heat
  5. SIFT(ASIFT) Matching with RANSAC
  6. Hadoop,Yarn,Zookeeper,kafka数据仓库集群命令集合
  7. python与vfp做桌面数据库_VFP数据库系统开发实例(附图)
  8. Python基础(三)--序列
  9. Windows7下硬盘安装RHEL 6.1
  10. (194)FPGA上电后IO的默认状态(ISE软件默认为0)
  11. JDBC学习笔记(查询SQL语句得到的结果对象)
  12. Python基础——@staticmethod与@classmethod
  13. 15个数据可视化的绝妙案例
  14. 【74HC595】STM32 74HC595驱动程序
  15. B样条曲线的公式推导及代码实现
  16. 交换字符使得字符串相同
  17. 有5个学生,每个学生的数据包括学号、姓名、三门课的成绩,从键盘输入5个学生数据,要求打印出三门课总平均成绩,以及最高分的学生的数据(包括学号、姓名、三门课的成绩、平均分数)。VS2019版
  18. mysql创建表s c sc_MySqL | 小白创建表
  19. Overload resolution ambiguity. All these functions match. kotlin 开发问题日常记录
  20. centos7如何安装视频播放器

热门文章

  1. 跨越跨域大山,前端不得不知道的Ajax
  2. c语言glut打正方形,OpenGL绘制正方形并用键盘移动
  3. [蓝桥杯][算法提高VIP]数的划分-dp
  4. 有序序列中的i个最大数(算法导论思考题9-1)
  5. java进阶之注解篇
  6. 递归函数中局部变量和全局变量
  7. Making the Grade POJ - 3666(离散化+dp)
  8. java8 stream 最大值_JDK8-Stream流常用方法
  9. 安装 java decompiler_Eclipse离线安装Java Decompiler插件(反编译)
  10. 哈希表创建哈希表(Hash Table,也叫散列表),是根据键(Key)而直接访问在内存存储位置的数据结构.typedef enum{ HASH_OK-icoding-数据结构-C