LeetCode 72. 编辑距离(中等)

题目描述

给你两个单词 word1word2请返回将 word1 转换成 word2 所使用的最少操作数

你可以对一个单词进行如下三种操作:

  • 插入一个字符
  • 删除一个字符
  • 替换一个字符

示例

示例 1:

复制代码
输入:word1 = "horse", word2 = "ros"
输出:3
解释:
horse -> rorse (将 'h' 替换为 'r')
rorse -> rose (删除 'r')
rose -> ros (删除 'e')

示例 2:

复制代码
输入:word1 = "intention", word2 = "execution"
输出:5
解释:
intention -> inention (删除 't')
inention -> enention (将 'i' 替换为 'e')
enention -> exention (将 'n' 替换为 'x')
exention -> exection (将 'n' 替换为 'c')
exection -> execution (插入 'u')

解法

动态规划

解题思路

定义mp[i][j] 表示将 word1 的前 i 个字符转换成 word2 的前 j 个字符所需的最少操作次数。

word1 = "horse" → word2 = "ros"

"" r o s
"" 0 1 2 3
h 1 1 2 3
o 2 2 1 2
r 3 2 2 2
s 4 3 3 2
e 5 4 4 3

以上面表格为例,mp的第一行和第一列的意义是word1和word2和""不断匹配过程,只能使用删除操作,所有每一次匹配,匹配数都要加1。

做好初始化后,需要判断word1[i - 1] == word2[j - 1]是否成立,如果成立,则此次匹配不需要操作,mp[i][j]和mp[i-1][j-1]就相等,否则,进行三种操作的比较,哪个操作总数最小,然后取最小。最终的mp[m][n]就是word1->word2的最少操作数。详细步骤请看代码注释。

cpp 复制代码
class Solution {
public:
    int minDistance(string word1, string word2) {
        int m = word1.size(), n = word2.size();
        
        // mp[i][j] 表示将 word1 的前i个字符转换为 word2 的前j个字符所需的最少操作次数
        vector<vector<int>> mp(m + 1, vector<int>(n + 1, 0));
        
        // 初始化第一行:空字符串"" 转换为 word2 的前j个字符
        for(int i = 1; i <= n; i++) {
            mp[0][i] = mp[0][i - 1] + 1;  // 每次在前一个基础上插入一个字符
        }
        
        // 初始化第一列:word1 的前i个字符转换为空字符串"",需要i次删除操作
        for(int i = 1; i <= m; i++) {
            mp[i][0] = mp[i - 1][0] + 1;  // 每次在前一个基础上删除一个字符
            
            // 填充当前行的其他列
            for(int j = 1; j <= n; j++) {
                // 检查当前字符是否匹配(注意索引要减1,因为dp表比字符串多一维)
                if(word1[i - 1] == word2[j - 1]) {
                    // 字符匹配:不需要额外操作,继承左上角的值
                    // 即:前i-1和前j-1个字符的编辑距离
                    mp[i][j] = mp[i - 1][j - 1];
                }
                else {
                    // 字符不匹配:取三种操作的最小值 + 1
                    // 1. mp[i-1][j] + 1:删除word1当前字符(先删除,再用前i-1个字符转换)
                    // 2. mp[i][j-1] + 1:插入word2当前字符(先用前i个字符转换前j-1个字符,再插入)
                    // 3. mp[i-1][j-1] + 1:替换word1当前字符为word2当前字符
                    mp[i][j] = min({mp[i - 1][j], mp[i][j - 1], mp[i - 1][j - 1]}) + 1;
                }
            }
        }
        
        // 返回将整个word1转换为整个word2所需的最少操作次数
        return mp[m][n];
    }
};

时间复杂度O(N^2),空间复杂度O(N^2)

相关推荐
Tisfy2 分钟前
LeetCode 3010.将数组分成最小总代价的子数组 I:排序 OR 维护最小次小
算法·leetcode·题解·排序·最小次小值
Learn Beyond Limits6 分钟前
文献阅读:A Probabilistic U-Net for Segmentation of Ambiguous Images
论文阅读·人工智能·深度学习·算法·机器学习·计算机视觉·ai
m0_7369191018 分钟前
编译器命令选项优化
开发语言·c++·算法
naruto_lnq25 分钟前
C++中的工厂方法模式
开发语言·c++·算法
千逐-沐风34 分钟前
SMU-ACM2026冬训周报2nd
算法
m0_748233171 小时前
C#与C语言:5大核心语法共性
java·jvm·算法
痴儿哈哈1 小时前
C++与硬件交互编程
开发语言·c++·算法
小O的算法实验室1 小时前
2024年ESWA SCI1区TOP,异构无人机配送问题的集成多目标优化方法,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
草履虫建模2 小时前
力扣算法 121. 买卖股票的最佳时机
算法·leetcode·职场和发展·贪心算法·动态规划·一次遍历
养军博客2 小时前
C语言五天速成(可用于蓝桥杯备考 难度中等偏下)
c语言·算法·蓝桥杯