94、动态规划-最长公共子序列

  1. 递归的基本思路
    • 比较两个字符串的最后一个字符。如果相同,则这个字符一定属于最长公共子序列,然后在剩余的字符串上递归求解。
    • 如果最后一个字符不相同,则分两种情况递归求解:
      • 去掉 text1 的最后一个字符,保留 text2 的所有字符。
      • 去掉 text2 的最后一个字符,保留 text1 的所有字符。
    • 取两种情况的最大值作为最终结果。
  2. 递归的终止条件
    • 如果任一字符串为空,则公共子序列长度为0。

代码如下:

public class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        return lcs(text1, text2, text1.length(), text2.length());
    }

    private int lcs(String text1, String text2, int i, int j) {
        // 终止条件:任一字符串长度为0
        if (i == 0 || j == 0) {
            return 0;
        }

        // 递归计算
        if (text1.charAt(i - 1) == text2.charAt(j - 1)) {
            // 最后一个字符相同,是LCS的一部分
            return 1 + lcs(text1, text2, i - 1, j - 1);
        } else {
            // 最后一个字符不同,选择不包含text1[i-1]或text2[j-1]的最大LCS
            return Math.max(lcs(text1, text2, i - 1, j), lcs(text1, text2, i, j - 1));
        }
    }
}

动态规划是优化递归的方法,使用表格来存储中间结果,避免重复计算。

  1. 定义状态

    • dp[i][j] 表示 text1[0...i-1]text2[0...j-1] 的最长公共子序列的长度。
  2. 状态转移方程

    • 如果 text1[i-1] == text2[j-1],则 dp[i][j] = dp[i-1][j-1] + 1
    • 如果 text1[i-1] != text2[j-1],则 dp[i][j] = max(dp[i-1][j], dp[i][j-1])
  3. 初始化

    • 初始化 dp 数组,当 ij 为0时,dp[i][j] = 0,表示一个字符串为空。
  4. 填表顺序

    • 由于每个 dp[i][j] 依赖于左边、上边和左上角的值,因此应从上到下、从左到右填表。
  5. 返回结果

    • dp[text1.length()][text2.length()] 即为两个字符串的最长公共子序列的长度。

代码如下:

public int longestCommonSubsequence(String text1, String text2) {
        // 获取输入字符串的长度
        int m = text1.length(), n = text2.length();
        
        // 创建动态规划表,多出的一行一列用于处理边界情况,即某一字符串长度为0的情况
        int[][] dp = new int[m + 1][n + 1];
        
        // 填充动态规划表
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                // 检查当前位置的字符是否相同
                if (text1.charAt(i - 1) == text2.charAt(j - 1)) {
                    // 如果当前两个字符相同,那么这两个字符属于LCS的一部分
                    // 因此dp[i][j]等于左上角的值加1
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                } else {
                    // 如果当前两个字符不相同,则LCS长度取决于不包含当前字符的子问题的最大值
                    // 即从左边或上边继承最大值
                    dp[i][j] = Math.max(dp[i - 1][j], dp[i][j - 1]);
                }
            }
        }
        
        // 返回最终结果,即整个字符串的最长公共子序列长度
        return dp[m][n];
    }

相关推荐

  1. [动态规划]公共序列

    2024-05-12 20:56:03       51 阅读
  2. 动态规划(DP)---- 公共序列

    2024-05-12 20:56:03       52 阅读
  3. 动态规划 Leetcode 1143 公共序列

    2024-05-12 20:56:03       39 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-05-12 20:56:03       98 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-05-12 20:56:03       106 阅读
  3. 在Django里面运行非项目文件

    2024-05-12 20:56:03       87 阅读
  4. Python语言-面向对象

    2024-05-12 20:56:03       96 阅读

热门阅读

  1. 【编写控制手机压测的脚本】

    2024-05-12 20:56:03       35 阅读
  2. 23种设计模式学习导航

    2024-05-12 20:56:03       32 阅读
  3. 【K8s】Kubectl 常用命令梳理

    2024-05-12 20:56:03       27 阅读
  4. 9. 学习distribute by rand()

    2024-05-12 20:56:03       28 阅读
  5. C语言从头学03——介绍函数printf

    2024-05-12 20:56:03       33 阅读
  6. [Easy] leetcode-225/232 栈和队列的相互实现

    2024-05-12 20:56:03       33 阅读
  7. 力扣 139. 单词拆分 python AC

    2024-05-12 20:56:03       32 阅读
  8. MATLAB数值计算工具箱介绍

    2024-05-12 20:56:03       34 阅读
  9. Linux 系统中,nl命令用于计算文件中的行号

    2024-05-12 20:56:03       34 阅读