动态规划Day33:编辑距离

392. 判断子序列

与最长公共子序列(LCS)的区别是,不能放弃s中的任何字符,所以当出现字符不匹配时,只能放弃t中的字符。

cpp 复制代码
    bool isSubsequence(string s, string t) {
        vector<vector<int>> dp(s.size() + 1, vector<int>(t.size() + 1, 0));
        for(int i = 1 ; i < s.size() + 1; i++){
            for(int j = 1; j < t.size() + 1; j++){
                if(s[i - 1] == t[j - 1]){
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                } else {
                    dp[i][j] = dp[i][j - 1];
                }
            }
        }
        if(dp[s.size()][t.size()] == s.size()){
            return true;
        }
        return false;
    }

115. 不同的子序列

dp[i][j]表示 字符串 s 的前 i 个字符(s[0..i-1])中,包含字符串 t 的前 j 个字符(t[0..j-1])的子序列个数

初始化

dp[i][0] = 1(对任意 i):t 的前 0 个字符(空字符串),是任何 s 的前 i 个字符的子序列(空序列只有 1 种选法:什么都不选);

dp[0][j] = 0(j>0):s 的前 0 个字符(空字符串),无法包含非空的 t 的前 j 个字符,个数为 0。

递推公式,s[i - 1] == t[j - 1]时,需要

递推公式

匹配:dp[i][j] = 用当前字符(dp[i-1][j-1]) + 不用当前字符(dp[i-1][j])

不匹配:dp[i][j] = 不用当前字符(dp[i-1][j])

cpp 复制代码
int numDistinct(string s, string t) {
        vector<vector<unsigned long long>> dp(s.size()+1, vector<unsigned long long>(t.size()+1));
        for(int i = 0; i < s.size()+1; i++){
            dp[i][0] = 1;
        }
        for(int j = 1; j < t.size()+1; j++){
            dp[0][j] = 0;
        }
        for(int i = 1; i < s.size()+1; i++){
            for(int j = 1; j < t.size()+1; j++){
                if(s[i - 1] == t[j - 1]){
                    dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];
                } else{
                    dp[i][j] = dp[i - 1][j];
                }
            }
        }
        return dp[s.size()][t.size()];
    }

583. 两个字符串的删除操作

可以转换成 寻找最大公共子序列,最后用两个字符串的长度分别减去最长公共子序列的长度,再相加就是答案

cpp 复制代码
int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1, 0));
        int res = 0;
        for(int i = 1; i < word1.size() + 1; i++){
            for(int j = 1; j < word2.size() + 1; j++){
                if(word1[i - 1] == word2[j - 1]){
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                } else {
                    dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
                }
                res = max(res, dp[i][j]);
            }
        }
        return word1.size() + word2.size() - res * 2;
    }

方法二

DP 定义:dp[i][j] 表示将 word1i 个字符转换为 word2j 个字符所需的最少操作数(仅含删除、插入)。

初始化:

  • dp[i][0] = iword2 为空时,需删除 word1i 个字符(共 i 次操作);

  • dp[0][j] = jword1 为空时,需插入 word2j 个字符(共 j 次操作)。

状态转移:

  • 字符匹配(word1[i-1] == word2[j-1]):无需操作,继承 dp[i-1][j-1]

  • 字符不匹配:取「删除 word1 当前字符(dp[i-1][j]+1)」或「插入 word2 当前字符(dp[i][j-1]+1)」的最小值。

cpp 复制代码
    int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1));
        for(int i = 0; i < word1.size() + 1; i++){
            dp[i][0] = i;
        }
        for(int j = 1; j < word2.size() + 1; j++){
            dp[0][j] = j;
        }
        for(int i = 1; i < word1.size() + 1; i++){
            for(int j = 1; j < word2.size() + 1; j++){
                if(word1[i - 1] == word2[j - 1]){
                    dp[i][j] = dp[i - 1][j - 1];
                } else {
                    dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1);
                }
            }
        }
        return dp[word1.size()][word2.size()];
    }

72. 编辑距离

DP 定义dp[i][j] 表示将 word1i 个字符转换为 word2j 个字符所需的最少操作数(支持插入、删除、替换三种操作)。

初始化

  • dp[i][0] = iword2 为空时,删除 word1i 个字符(共 i 次操作);

  • dp[0][j] = jword1 为空时,插入 word2j 个字符(共 j 次操作)。

状态转移

  • 字符匹配:无需操作,直接继承 dp[i-1][j-1]

  • 字符不匹配:取「删除(dp[i-1][j]+1)、删除(dp[i][j-1]+1)、替换(dp[i-1][j-1]+1)」三者最小值。

删除word1 ,就是增加word2,所以对两个都执行山删除操作可以视为其中一次删除是对另一个的增加

结果dp[word1.size()][word2.size()] 即为转换的最少操作数。

cpp 复制代码
    int minDistance(string word1, string word2) {
        vector<vector<int>> dp(word1.size() + 1, vector<int> (word2.size() + 1));
        for(int i = 0; i < word1.size() + 1; i++){
            dp[i][0] = i;
        }
        for(int j = 1; j < word2.size() + 1; j++){
            dp[0][j] = j;
        }
        for(int i = 1; i < word1.size() + 1;i++){
            for(int j = 1; j < word2.size() + 1; j++){
                if(word1[i - 1] == word2[j - 1]){
                    dp[i][j] = dp[i - 1][j - 1];
                } else {
                    dp[i][j] = min( min(dp[i - 1][j] + 1, dp[i][j - 1] + 1), dp[i - 1][j - 1] + 1);
                }
            }
        }
        return dp[word1.size()][word2.size()];
    }
相关推荐
每天要多喝水1 小时前
动态规划Day34:回文
算法·动态规划
weixin_477271692 小时前
马王堆帛书《周易》系统性解读(《函谷门》原创)
算法·图搜索算法
AomanHao2 小时前
【ISP】基于暗通道先验改进的红外图像透雾
图像处理·人工智能·算法·计算机视觉·图像增强·红外图像
We་ct2 小时前
LeetCode 226. 翻转二叉树:两种解法(递归+迭代)详解
前端·算法·leetcode·链表·typescript
大熊背2 小时前
APEX系统中为什么 不用与EV0的差值计算曝光参数调整量
人工智能·算法·apex·自动曝光
REDcker2 小时前
HDR Vivid 技术介绍
数据库·算法·视频·sdr·屏幕·显示技术·dhr
ab1515172 小时前
2.18完成109、112、113
算法
追随者永远是胜利者3 小时前
(LeetCode-Hot100)64. 最小路径和
java·算法·leetcode·职场和发展·go
汉克老师3 小时前
GESP2024年6月认证C++二级( 第三部分编程题(1) 平方之和)
c++·算法·预处理·完全平方数·循环结构·gesp二级·gesp2级