问题

求一个字符串有多少个回文子串

Input: "abc"

Output: 3

Input: "aaa"

Output: 6

思路和代码(1)——朴素做法

用dp的基本思想“子问题求解”,但是因为没有重叠子问题,没必要用dp数组或者矩阵来存储中间结果。

直接遍历每个字符,然后以该字符为中心向两边扩张来判断是否回文串,是的话就总数加1。

这时要考虑两种情况,一种是以单字符为中心来扩张即aba这种,另一种是以双字符为中心来扩张即abba。

对于每个字符s[i]要遍历两次扩张,一次以s[i]为中心来扩张,一次以s[i]和s[i+1]为中心来扩张。因此需要遍历2n次(n为字符串长度)。又考虑到最后一个字符s[n-1]后面已经没有字符了,不需要考虑以两个字符为中心的情况,所以实际是遍历2n-1次。

可以用两个指针left和right,单字符为中心的扩张时left和right都初始指向s[i],双字符为中心的扩张时left指向s[i],right指向s[i+1]。

时间复杂度O(n^2),空间复杂度O(1)

class Solution(object):
def countSubstrings(self, s):
"""
:type s: str
:rtype: int
"""
n = len(s)
res = 0
for i in range(2*n-1):
left = i/2
right = left + i%2
while(left >= 0 and right < n and s[left]==s[right] ):
res += 1
left -= 1
right += 1
return res

思路和代码(2)——Manacher算法

使用马拉车算法,算法根据回文串的对称特点利用之前的信息计算,可以在线性时间内找出所有回文串。

“原字符串”转为“#字符串”

前面说了回文串分为单字符中心和双字符中心,为了可以统一按单字符中心处理。首先在原字符中插入字符'#',例如aaa经过插入处理后得到"#a#a#a#",这样遍历到第一个a时其实是以a为单字符中心,遍历到第二个#时就是以aa为双字符中心。第一个#和最后一个#的意义在于补全第一个字符和最后一个字符的“回文串身份”,因为单个字符也算回文串。

镜像计算

我们用p[i]表示以i为中心的回文串半径,p[i]初始化为0。正常情况,我们根据扩张的方法来判断半径是否增加。

特殊情况,如果字符s[i]处于某个已知的“以id为中心,以mx为右边界”的回文串中(此时mx > i),我们称这个回文串为id回文串,此时可以利用id为中心对称过去的镜像索引来简化计算,我们称镜像索引为j。然后有两种情况。

如果p[j]<mx-i,说明i的半径没有超出mx的边界而且等于j的半径,此时p[i]=[j]=p[2*id-1],这个p[i]值是确定的。

如果p[j]>=mx-i,说明i的半径至少扩张到了mx,mx之后的部分要继续扩张。此时p[i]值不是确定的,但是我们让p[i]=mx-i,因为半径为mx-i是至少的,至于半径是否继续增加取决于mx后面的部分,我们在扩张步骤计算。

简化以上两种情况可以得到:如果mx > i,那么p[i] = min(p[2*id-1], mx-i)

扩张

根据上面的说明,只有在mx >i 以及 p[j] < mx-i的情况下可以通过镜像计算来直接获得半径,其它情况还是要继续扩张。扩张的计算很简单,在当前半径的基础上扩张,判断s[i+p[i]+1]和s[i-p[i]-1]是否相等,相等则半径p[i]加1。

更新id回文串

遍历过程中,如果发现有右边界更大的回文串,则覆盖这个回文串。

“#字符串”的回文个数->“原字符串”的回文个数

上面计算的半径p[i]会因为#的插入而增多,所以要根据“#字符串”的半径计算一下,得到“原字符串”的半径。我们把所有半径加起来就可以计算出回文字串的数量。

举个双字符为中心的例子,#a#a#,中间#的半径为a#,要转为半径a,2要转为1,直接除以2就得到了。

举个单字符为中心的例子,#a#a#b#a#a#,b的半径#a#a#,要转为半径aa,5要转为2,但是考虑b本身也是一个回文串,实际上是5要转为3,这个时候需要加1之后除以2(这里跟半径的定义有关,我这里的定义中b这个单字符的半径为0。如果定义的b的半径为1,这里的计算就不是加1而是减1了)。

综合来看,直接加1除以2就可以了,因为/会向下取整。

时间复杂度O(n),空间复杂度O(1)

class Solution(object):
def countSubstrings(self, s):
"""
:type s: str
:rtype: int
"""
s = '#' + '#'.join(s) + '#'
n = len(s)
p = [0]*n
id = mx = res = 0
for i in range(1,n-1):
if(mx > i):
p[i] = min(mx-i, p[2*id-i])
while i+p[i]+1 < n and i-p[i]-1 >=0 and s[i+p[i]+1] == s[i-p[i]-1]:
p[i] += 1
if(i + p[i] > mx):
id = i
mx = i + p[i]
res += (p[i]+1)/2
return res

647. Palindromic Substrings(马拉车算法)的更多相关文章

  1. 【LeetCode】647. Palindromic Substrings 解题报告(Python)

    [LeetCode]647. Palindromic Substrings 解题报告(Python) 标签: LeetCode 题目地址:https://leetcode.com/problems/p ...

  2. Leetcode 647. Palindromic Substrings

    Given a string, your task is to count how many palindromic substrings in this string. The substrings ...

  3. 2015 UESTC 搜索专题M题 Palindromic String 马拉车算法

    Palindromic String Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://acm.uestc.edu.cn/#/contest/s ...

  4. 647. Palindromic Substrings 互文的子字符串

    [抄题]: Given a string, your task is to count how many palindromic substrings in this string. The subs ...

  5. [LeetCode] 647. Palindromic Substrings 回文子字符串

    Given a string, your task is to count how many palindromic substrings in this string. The substrings ...

  6. 647. Palindromic Substrings

    Given a string, your task is to count how many palindromic substrings in this string. The substrings ...

  7. 【Leetcode】647. Palindromic Substrings

    Description Given a string, your task is to count how many palindromic substrings in this string. Th ...

  8. LeetCode 647. Palindromic Substrings的三种解法

    转载地址 https://www.cnblogs.com/AlvinZH/p/8527668.html#_label5 题目详情 给定一个字符串,你的任务是计算这个字符串中有多少个回文子串. 具有不同 ...

  9. 【LeetCode】647. Palindromic Substrings 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 方法一:暴力循环 方法二:固定起点向后找 方法三:动 ...

随机推荐

  1. C++引用具体解释

    引用是C++中新出现的.有别于C语言的语法元素之中的一个. 关于引用的说明,网络上也有不少.可是总感觉云遮雾绕,让人印象不深刻. 今天我就来深入解释一下引用.并就一些常见的观点进行说明,最后附带代码演 ...

  2. Spring_day04--HibernateTemplate介绍_整合其他方式_Spring分模块开发

    HibernateTemplate介绍 1 HibernateTemplate对hibernate框架进行封装, 直接调用HibernateTemplate里面的方法实现功能 2 HibernateT ...

  3. springboot整合mybatis之用外置服务器启动项目(二)

    在上一篇中我们是用的springboot自带的tomcat服务器,接下来想试一下 将springboot当做一个web项目 放在eclipse中用tomcat来启动. 首先在pom.xml中加上,移除 ...

  4. OOXML,XLSX分析

    07以上的xlsx是使用了OOXML和zip,将后缀修改为.zip,就可以看到文件,主要分析xl目录下的文件,如图: 主要数据文件在xl目录下面 styles.xml里面存放着excel的样式数据 很 ...

  5. 微软笔试题-highways

    题目大意 一条单向的高速公路上有N辆车,在0时刻,每辆车分别在起点A[0],A[1]....处开始从北向南出发,每辆车有个终点B[0],B[1]....且每辆车有个限制速度 V[0],V[1]... ...

  6. python练习题-3

    author:headsen chen date: 2018-06-01  15:51:05 习题 31:  作出决定(if + raw_input) [root@localhost py]# cat ...

  7. Android遍历SqlLite cursor对象:

    //1. Cursor c =...; for(c.moveToFirst(); ! c.isAfterLast(); c.moveToNext()){ //c… } //2. Cursor curs ...

  8. 解决多指操作放大缩小 指针错误 java.lang.IllegalArgumentException: pointerIndex out of range

    /** Custom your own ViewPager to extends support ViewPager. java source: */ /** Created by azi on 20 ...

  9. 06.Curator Barrier

        分布式Barrier是这样一个类: 它会阻塞所有节点上的等待进程,知道某一个被满足, 然后所有的节点继续进行.     比如赛马比赛中, 等赛马陆续来到起跑线前. 一声令下,所有的赛马都飞奔而 ...

  10. [iOS微博项目 - 4.2] - 设置转发微博背景

    github: https://github.com/hellovoidworld/HVWWeibo A.转发微博部分的淡灰色背景 1.需求 转发微博部分需要设置背景色 使用图片作为背景   2.思路 ...