最长回文子序列

8/12/2021 DPLeetCode

给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。

# 题目描述

给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。

子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。

516. 最长回文子序列 (opens new window)

# 示例

输入:s = "bbbab"
输出:4
解释:一个可能的最长回文子序列为 "bbbb" 。

输入:s = "cbbd"
输出:2
解释:一个可能的最长回文子序列为 "bb" 。
1
2
3
4
5
6
7

# 提示

  • 1 <= s.length <= 1000
  • s 仅由小写英文字母组成

# 解法:DP

  • 时间复杂度:O(n2)\mathcal{O}(n^2)
  • 空间复杂度:O(n2)\mathcal{O}(n^2)

对于一个子序列而言,如果它是回文子序列,并且长度大于 2,那么将它首尾的两个字符去除之后,它仍然是个回文子序列。因此可以用动态规划的方法计算给定字符串的最长回文子序列。

func longestPalindromeSubseq(s string) int {
    var dp []int
    for i := len(s) - 1; i >= 0; i-- {
        nxtDp := make([]int, len(s) - i)
        for j := i; j < len(s); j++ {
            if s[i] == s[j] {
                if j - i <= 1 {
                    nxtDp[j - i] = j - i + 1 
                } else {
                    nxtDp[j - i] = dp[j - i - 2] + 2
                }
            } else {
                nxtDp[j - i] = max(dp[j - i - 1], nxtDp[j - i - 1]) 
            }
        }
        dp = nxtDp
    }
    return dp[len(s) - 1]
}

func max(a, b int) int {
    if a > b {
        return a
    }
    return b
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
上次更新于: a year ago