力扣 583. 两个字符串的删除操作(中等)

题目

给定两个单词 word1 和 word2,找到使得 word1 和 word2 相同所需的最小步数,每步可以删除任意一个字符串中的一个字符。
示例:

输入: "sea", "eat"
输出: 2
解释: 第一步将"sea"变为"ea",第二步将"eat"变为"ea"

提示:

  • 给定单词的长度不超过500。
  • 给定单词中的字符只含有小写字母。

题解

方法一:最长公共子序列
将问题转换为找两个字符串的最长公共子序列,那么结果为两个字符串的长度和减去最长公共子序列的长度*2,这就是删除的最小步数。
给定两个字符串 word 1 \textit{word}_1 word1 word 2 \textit{word}_2 word2,分别删除若干字符之后使得两个字符串相同,则剩下的字符为两个字符串的公共子序列。为了使删除操作的次数最少,剩下的字符应尽可能多。当剩下的字符为两个字符串的最长公共子序列时,删除操作的次数最少。因此,可以计算两个字符串的最长公共子序列的长度,然后分别计算两个字符串的长度和最长公共子序列的长度之差,即为两个字符串分别需要删除的字符数,两个字符串各自需要删除的字符数之和即为最少的删除操作的总次数。

class Solution {
    public int minDistance(String word1, String word2) {
        int m = word1.length();
        int n = word2.length();
        int[][] dp = new int[m + 1][n + 1];   // dp[i][j] 表示word1[0,...,i-1]与word2[0,...,j-1]的最长公共子序列的长度
        for(int i = 1; i <= m; i++){
            for(int j = 1; j <= n; j++){
                if(word1.charAt(i - 1) == word2.charAt(j - 1)){
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                }else{
                    dp[i][j] = Math.max(dp[i - 1][j], dp[i][j-1]);
                }
            }
        }
        return m + n - 2 * dp[m][n];

    }
}

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