You may be out, but you never lose the attitude.

人可以受挫,但精神不可以倒下。

问题描述

给定一个字符串s,找到其中最长的回文子序列,并返回该序列的长度。可以假设s的最大长度为 1000 。

 

示例 1:
输入:

"bbbab"

输出:

4

一个可能的最长回文子序列为 "bbbb"。

 

示例 2:
输入:

"cbbd"

输出:

2

一个可能的最长回文子序列为 "bb"。

 

提示:

  • 1 <= s.length <= 1000

  • s 只包含小写英文字母

 

动态规划解决

注意这里是求最长回文子序列不是回文子串,子串必须是连续的,但子序列不一定是连续的。

 

我们定义dp[i][j]表示字符串中从i到j之间的最长回文子序列。

1,如果s.charAt(i) == s.charAt(j),也就是说两头的字符是一样的,他们可以和中间的最长回文子序列构成一个更长的回文子序列,即

dp[i][j]=dp[i+1][j-1]+2

如下图所示

529,动态规划解最长回文子序列_最长回文子序列

2,如果s.charAt(i) != s.charAt(j),说明i和j指向的字符是不相等的,我们可以截取,要么去掉i指向的字符,要么去掉j指向的字符,然后取最大值,即

dp[i][j] = Math.max(dp[i][j-1], dp[i+1][j])

如下图所示

 

529,动态规划解最长回文子序列_编程开发_02

有了递推公式,我们再来看下Base case,就是一个字符也是回文串,即dp[i][i]=1;

注意二维数组dp[i][j]的定义,如果i>j是没有意义的。再来看一下他们之间的关系

529,动态规划解最长回文子序列_编程开发_03

从上面图中可以看出如果我们想求dp[i][j],那么其他3个必须都是已知的,很明显从上往下遍历是不行的,我们只能让i从最后一个字符往前遍历,j从i的下一个开始遍历,最后只需要返回dp[0][length - 1]即可。来看下代码

 1public int longestPalindromeSubseq(String s) {
2    int length = s.length();
3    int[][] dp = new int[length][length];
4    //这里i要从最后一个开始遍历
5    for (int i = length - 1; i >= 0; i--) {
6        //单个字符也是一个回文串
7        dp[i][i] = 1;
8        //j从i的下一个开始
9        for (int j = i + 1; j < length; j++) {
10            //下面是递推公式
11            if (s.charAt(i) == s.charAt(j)) {
12                dp[i][j] = dp[i + 1][j - 1] + 2;
13            } else {
14                dp[i][j] = Math.max(dp[i + 1][j], dp[i][j - 1]);
15            }
16        }
17    }
18    return dp[0][length - 1];
19}

 

总结

这里递推公式比较好找,关键点在于字符串遍历的顺序。

 

529,动态规划解最长回文子序列_最长回文子序列_04

515,动态规划解买卖股票的最佳时机含手续费

493,动态规划解打家劫舍 III

486,动态规划解最大子序和

413,动态规划求最长上升子序列

 

 

截止到目前我已经写了500多道算法题了,为了方便大家阅读,我把部分算法题整理成了pdf文档,目前有900多页,大家可以在公众号中回复关键字“pdf”即可获取下载链接。

 

529,动态规划解最长回文子序列_编程开发_05你点的每个赞,我都认真当成了喜欢