动态规划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()];
    }
相关推荐
吞下星星的少年·-·5 分钟前
区间转化、扫描线
算法
笨笨饿16 分钟前
80_聊聊SPI以及它们的变体
linux·c语言·网络·stm32·单片机·算法·个人开发
Ricardo-Yang17 分钟前
使用GEE以及LandSat8植被指数NDVI计算
python·深度学习·神经网络·算法·视觉检测
sheeta199822 分钟前
LeetCode 每日一题笔记 日期:2026.05.12 题目:1665. 完成所有任务的最少初始能量
笔记·算法·leetcode
khalil102023 分钟前
代码随想录算法训练营Day-49 图论01 | 图论理论基础、深搜理论基础、98. 所有可达路径、广搜理论基础
c++·算法·leetcode·深度优先·图论
Chase_______32 分钟前
【算法】删除子数组的最大得分 & 最多 K 个重复元素的最长子数组——不定长滑动窗口与哈希频率约束
算法·哈希算法
fie888944 分钟前
城市环境下车辆目标跟踪算法 MATLAB 实现
算法·matlab·目标跟踪
tryCbest1 小时前
软考 - 排序算法
算法·排序算法
AKA__Zas1 小时前
芝士算法(双指针篇 1.0)
java·算法·学习方法
吃着火锅x唱着歌1 小时前
LeetCode 726.原子的数量
linux·算法·leetcode