剑指 Offer II 095. 最长公共子序列(中等 动态规划 字符串)

剑指 Offer II 095. 最长公共子序列

给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。

一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

例如,“ace” 是 “abcde” 的子序列,但 “aec” 不是 “abcde” 的子序列。
两个字符串的 公共子序列 是这两个字符串所共同拥有的子序列。

示例 1:

输入:text1 = “abcde”, text2 = “ace”
输出:3
解释:最长公共子序列是 “ace” ,它的长度为 3 。
示例 2:

输入:text1 = “abc”, text2 = “abc”
输出:3
解释:最长公共子序列是 “abc” ,它的长度为 3 。
示例 3:

输入:text1 = “abc”, text2 = “def”
输出:0
解释:两个字符串没有公共子序列,返回 0 。

提示:

1 <= text1.length, text2.length <= 1000
text1 和 text2 仅由小写英文字符组成。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/qJnOS7
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

分析

方法一:空间复杂度O(mn)
1、寻找状态转移方程。与之前的题目不同的是,该题有两个输入,我们假设f(i , j)表示第一个字符串索引 0 到 i 与第二个字符串索引 0 到 j 的最长公共子序列,那么f(i , j)与索引 i 和 j 之前的f(i - 1 , j - 1)以及f(i - 1 , j)、f(i , j - 1)之间有什么关系呢?如果字符串s1的索引 i 和字符串s2的索引 j 处的字符相同,那么显而易见f(i , j) = f(i - 1 , j - 1) + 1,相当于在i - 1和j - 1的公共子序列后面添加一个字符,如果不相同,那么这两个字符不可能同时出现在公共子序列中,这时f(i , j)的最长公共子序列要么是 0 到 i - 1 与 0 到 j 的公共子序列,要么是 0 到 i 与 0 到 j - 1 的公共子序列,即f(i , j) = max(f(i - 1, j), f(i , j - 1))。特殊地,i或者j等于0时,i - 1和j - 1代表空字符串,则 f(-1, j) = 0,f(i, -1) = 0。
2、根据状态转移方程求dp数组。考虑到索引 -1 的情况,设两个字符串的长度分别为m,n,dp为(m + 1) * (n + 1)的二维数组。嵌套遍历两个字符串填充dp数组,最后返回dp[m][n]即可。
方法二:空间复杂度O(mn)的优化
f(i, j)只依赖于 f(i - 1 , j - 1) 和 f(i - 1, j)、 f(i , j - 1)的值,所以dp数组只保存两行值即可。
方法三:空间复杂度O(min(m, n))
只用一个一维数组保存所有计算所需的信息,一维数组的长度是字符串短的那个长度加一。为了让一个一维数组保存之前的两行信息,一维数组的每个位置需要保存原来每一列的两个信息,即原来的f(i, j)和f(i - 1, j)都保存在dp[j + 1]的位置,在获得最后的dp[j + 1]之前,dp[j + 1]保存的是f(i - 1, j)的值,在完成最后的计算后dp[j + 1]替换成最终的目标值。
因为在完成最后的计算之前可能还需要f(i - 1, j)的值,所以要把同时先保存起来。在题解的代码中,prev用于保存数组中被替换的值,在计算f(i, j)之前prev保存的是f(i - 1, j - 1)的值,在计算f(i, j)之后,把原来dp[j + 1]位置的f(i - 1, j)保存在prev中,同时把f(i, j)保存在dp[j + 1]中,下一步计算f(i, j + 1)时可以从prev中得到f(i - 1, j)。
在代码cur = Math.max(dp[j], dp[j + 1])中,dp[j]对应的是f(i, j - 1),dp[j + 1]对应的是f(i - 1, j)。由于遍历是按照从上到下和从左到右的,所以计算f(i, j)之前f(i, j - 1)已经计算出来并保存在dp[j]的位置了,f(i - 1, j)还是dp[j + 1]处保存的上次遍历(i - 1)的值。

题解(Java)

方法一:空间复杂度O(mn)

class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        int len1 = text1.length(), len2 = text2.length();
        int[][] dp = new int[len1 + 1][len2 + 1];
        for (int i = 0; i < len1; i++) {
            for (int j = 0; j < len2; j++) {
                if (text1.charAt(i) == text2.charAt(j)) {
                    dp[i + 1][j + 1] = dp[i][j] + 1;
                } else {
                    dp[i + 1][j + 1] = Math.max(dp[i + 1][j], dp[i][j + 1]);
                }
            }
        }
        return dp[len1][len2];
    }
}

方法二:空间复杂度O(mn)的优化

class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        int len1 = text1.length(), len2 = text2.length();
        if (len1 < len2) {
            return longestCommonSubsequence(text2, text1);
        }
        int[][] dp = new int[2][len2 + 1];
        for (int i = 0; i < len1; i++) {
            for (int j = 0; j < len2; j++) {
                if (text1.charAt(i) == text2.charAt(j)) {
                    dp[(i + 1) % 2][j + 1] = dp[i % 2][j] + 1;
                } else {
                    dp[(i + 1) % 2][j + 1] = Math.max(dp[(i + 1) % 2][j], dp[i % 2][j + 1]);
                }
            }
        }
        return dp[len1 % 2][len2];
    }
}

方法三:空间复杂度O(min(m, n))

class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        int len1 = text1.length(), len2 = text2.length();
        if (len1 < len2) {
            return longestCommonSubsequence(text2, text1);
        }
        int[] dp = new int[len2 + 1];
        for (int i = 0; i < len1; i++) {
            int prev = dp[0];
            for (int j = 0; j < len2; j++) {
                int cur;
                if (text1.charAt(i) == text2.charAt(j)) {
                    cur = prev + 1;
                } else {
                    cur = Math.max(dp[j], dp[j + 1]);
                }
                prev = dp[j + 1];
                dp[j + 1] = cur;
            }
        }
        return dp[len2];
    }
}

你可能感兴趣的:(动态规划,算法,leetcode)