跳转至

14.6 编辑距离问题

编辑距离,也称 Levenshtein 距离,指两个字符串之间互相转换的最少修改次数,通常用于在信息检索和自然语言处理中度量两个序列的相似度。

Question

(输入两个字符串 ,返回将 转换为 所需的最少编辑步数。)

你可以在一个字符串中进行三种编辑操作:插入一个字符、删除一个字符、将字符替换为任意一个字符。

如图 14-27 所示,将 kitten 转换为 sitting 需要编辑 3 步,包括 2 次替换操作与 1 次添加操作;将 hello 转换为 algo 需要 3 步,包括 2 次替换操作和 1 次删除操作。

编辑距离的示例数据

图 14-27 编辑距离的示例数据

编辑距离问题可以很自然地用决策树模型来解释 。字符串对应树节点,一轮决策(一次编辑操作)对应树的一条边。

如图 14-28 所示,在不限制操作的情况下,每个节点都可以派生出许多条边,每条边对应一种操作,这意味着从 hello 转换到 algo 有许多种可能的路径。

从决策树的角度看,本题的目标是求解节点 hello 和节点 algo 之间的最短路径。

基于决策树模型表示编辑距离问题

图 14-28 基于决策树模型表示编辑距离问题

动态规划思路

第一步:思考每轮的决策,定义状态,从而得到

每一轮的决策是对字符串 进行一次编辑操作。

我们希望在编辑操作的过程中,问题的规模逐渐缩小,这样才能构建子问题。设字符串 的长度分别为 ,我们先考虑两字符串尾部的字符

  • 相同,我们可以跳过它们,直接考虑
  • 不同,我们需要对 进行一次编辑(插入、删除、替换),使得两字符串尾部的字符相同,从而可以跳过它们,考虑规模更小的问题。

也就是说,我们在字符串 中进行的每一轮决策(编辑操作),都会使得 中剩余的待匹配字符发生变化。因此,状态为当前在 中考虑的第 和第 个字符,记为

状态 对应的子问题: 的前 个字符更改为 的前 个字符所需的最少编辑步数

至此,得到一个尺寸为 的二维 表。

第二步:找出最优子结构,进而推导出状态转移方程

考虑子问题 ,其对应的两个字符串的尾部字符为 ,可根据不同编辑操作分为图 14-29 所示的三种情况。

  1. 之后添加 ,则剩余子问题
  2. 删除 ,则剩余子问题
  3. 替换为 ,则剩余子问题

编辑距离的状态转移

图 14-29 编辑距离的状态转移

根据以上分析,可得最优子结构: 的最少编辑步数等于 三者中的最少编辑步数,再加上本次的编辑步数 。对应的状态转移方程为:

请注意, 相同时,无须编辑当前字符 ,这种情况下的状态转移方程为:

第三步:确定边界条件和状态转移顺序

当两字符串都为空时,编辑步数为 ,即 。当 为空但 不为空时,最少编辑步数等于 的长度,即首行 。当 不为空但 为空时,最少编辑步数等于 的长度,即首列

观察状态转移方程,解 依赖左方、上方、左上方的解,因此通过两层循环正序遍历整个 表即可。

代码实现

/* 编辑距离:空间优化后的动态规划 */
int editDistanceDPComp(string s, string t) {
    int n = s.length(), m = t.length();
    vector<int> dp(m + 1, 0);
    // 状态转移:首行
    for (int j = 1; j <= m; j++) {
        dp[j] = j;
    }
    // 状态转移:其余行
    for (int i = 1; i <= n; i++) {
        // 状态转移:首列
        int leftup = dp[0]; // 暂存 dp[i-1, j-1]
        dp[0] = i;
        // 状态转移:其余列
        for (int j = 1; j <= m; j++) {
            int temp = dp[j];
            if (s[i - 1] == t[j - 1]) {
                // 若两字符相等,则直接跳过此两字符
                dp[j] = leftup;
            } else {
                // 最少编辑步数 = 插入、删除、替换这三种操作的最少编辑步数 + 1
                dp[j] = min(min(dp[j - 1], dp[j]), leftup) + 1;
            }
            leftup = temp; // 更新为下一轮的 dp[i-1, j-1]
        }
    }
    return dp[m];
}