【Hot 100 刷题计划】 LeetCode 72. 编辑距离 | C++ 经典 DP 增删改状态转移

LeetCode 72. 编辑距离

📌 题目描述

题目级别:困难 (大厂极高频)

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

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

  1. 插入一个字符
  2. 删除一个字符
  3. 替换一个字符
  • 示例 1:
    输入:word1 = "horse", word2 = "ros"
    输出:3
    解释:
    horse -> rorse (将 'h' 替换为 'r')
    rorse -> rose (删除 'r')
    rose -> ros (删除 'e')

💡 破题思路:二维动态规划

编辑距离是字符串 DP 的终极 Boss,但只要拆解开它的状态转移,逻辑其实非常清晰。

状态定义:

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

状态转移推导:

当我们考察 word1 的第 i 个字符和 word2 的第 j 个字符时,我们有三种基础操作路线:

  1. 删除 :假设我们已经把 word1 的前 i-1 个字符变成了 word2 的前 j 个字符,那么我们只需要把 word1 多出来的第 i 个字符删除 即可。代价:f[i - 1][j] + 1
  2. 插入 :假设我们已经把 word1 的前 i 个字符变成了 word2 的前 j-1 个字符,那么我们只需要再插入 一个 word2 的第 j 个字符即可。代价:f[i][j - 1] + 1
  3. 替换/跳过 :假设我们已经把 word1 的前 i-1 变成了 word2 的前 j-1
    • 如果当前字符相等(word1[i] == word2[j]),不需要操作,直接跳过。代价:f[i - 1][j - 1] + 0
    • 如果当前字符不等,我们需要做替换 操作。代价:f[i - 1][j - 1] + 1

我们取这三条路线中的最小值即可!

本解法高光点(哨兵技巧与边界处理):

代码中使用了 word1 = " " + word1; 加入空格哨兵,让真实字符从下标 1 开始,完美避免了 i-1 越界。

同时,必须正确初始化边界:

  • f[i][0] = i:把长度为 i 的字符串变成空串,必须删除 i 次。
  • f[0][j] = j:把空串变成长度为 j 的字符串,必须插入 j 次。

💻 C++ 代码实现 (标准二维 DP)

cpp 复制代码
class Solution {
public:
    int minDistance(string word1, string word2) {
        int m = word1.size(), n = word2.size();
        
        // 极客技巧:头部垫一个空格,让后续有效字符下标从 1 开始
        word1 = " " + word1;
        word2 = " " + word2;

        int f[m + 10][n + 10];
        // 初始化为一个较大值
        memset(f, 0x3f3f3f3f, sizeof f);

        // 边界初始化极其重要!
        // word1 的前 i 个字符变为空串,需要删除 i 次
        for (int i = 0; i <= m; i++) f[i][0] = i;
        // 空串变成 word2 的前 j 个字符,需要插入 j 次
        for (int j = 0; j <= n; j++) f[0][j] = j;

        // 遍历填写二维表格
        for (int i = 1; i <= m; i ++ )
        {
            for (int j = 1; j <= n; j ++ )
            {
                // 路线 1 & 2:从上方下来(删除),或从左方过来(插入)
                f[i][j] = min(f[i - 1][j], f[i][j - 1]) + 1;
                
                // 路线 3:从左上角过来(替换或直接匹配跳过)
                // 因为前面垫了空格,这里的下标直接用 i 和 j
                int cost = (word1[i] == word2[j] ? 0 : 1);
                f[i][j] = min(f[i][j], f[i - 1][j - 1] + cost);
            }
        }

        return f[m][n];
    }
};
相关推荐
穿条秋裤到处跑1 小时前
每日一道leetcode(2026.04.16):距离最小相等元素查询
算法·leetcode·职场和发展
无限进步_2 小时前
【C++】寻找字符串中第一个只出现一次的字符
开发语言·c++·ide·windows·git·github·visual studio
楼田莉子2 小时前
Linux网络:IP协议
linux·服务器·网络·c++·学习·tcp/ip
wuminyu2 小时前
专家视角看JVM_StartThread
java·linux·c语言·jvm·c++
敲上瘾2 小时前
高并发内存池(三):PageCache(页缓存)的实现
linux·c++·缓存·高并发内存池·池化技术
XY_墨莲伊2 小时前
【实战项目】基于B/S结构Flask+Folium技术的出租车轨迹可视化分析系统(文末含完整源代码)
开发语言·后端·python·算法·机器学习·flask
小雅痞2 小时前
[Java][Leetcode simple] 1. 两数之和
java·算法·leetcode
somi73 小时前
ARM-驱动-09-LCD FrameBuffer
arm开发·驱动开发·算法·自用
乐迪信息3 小时前
乐迪信息:智慧港口AI防爆摄像机实现船舶违规靠岸自动抓拍
大数据·人工智能·算法·安全·目标跟踪