找回密码
 立即注册
查看: 31|回复: 0

洛谷P2758题解:动态规划求解编辑距离的完整攻略

[复制链接]
  • 打卡等级:即来则安
  • 打卡总天数:28
  • 打卡月天数:2
  • 打卡总奖励:194
  • 最近打卡:2025-08-15 15:43:34

36

主题

2

回帖

2544

积分

VIP会员

积分
2544
发表于 2025-7-29 11:06:37 | 显示全部楼层 |阅读模式
一、题目解读
洛谷P2758题要求计算两个字符串之间的编辑距离,即通过插入、删除、替换三种操作将字符串A转换为B所需的最小操作次数。题目考察的核心是动态规划算法字符串匹配中的应用,需要找到**的状态转移路径。
二、解题思路
采用动态规划(Dynamic Programming)策略。核心思想是构建二维dp数组,dp[j]表示A的前i个字符转换为B的前j个字符的最小操作次数。通过初始化边界条件(空串转换的情况)和状态转移方程(字符相同时无需操作,不同时选择插入、删除、替换中的**解),最终得到dp[m][n]即为答案。
三、解题步骤
1. 输入与初始化:读取字符串A和B,记录长度m和n,创建dp数组。
2. 边界条件:dp[0] = i(删除A前i个字符),dp[0][j] = j(插入B前j个字符)。
3. 状态转移:
    若A[i-1] == B[j-1],则dp[j] = dp[i-1][j-1](无需操作)。
    否则,计算三种操作的最小值:插入(dp[j-1] + 1)、删除(dp[i-1][j] + 1)、替换(dp[i-1][j-1] + 1)。
4. 输出结果:dp[m][n]即为最小编辑距离。
四、代码与注释
  1. #include <iostream>
  2. #include <string>
  3. #include <algorithm>
  4. using namespace std;

  5. int main() {
  6.     string A, B;
  7.     cin >> A >> B;   // 输入两个字符串
  8.     int m = A.length(), n = B.length();
  9.    
  10.     // dp[i][j]表示A的前i个字符转换为B的前j个字符的最小操作次数
  11.     int dp[m+1][n+1];
  12.    
  13.     // 初始化边界条件
  14.     for(int i = 0; i <= m; i++) dp[i][0] = i; // 删除i次
  15.     for(int j = 0; j <= n; j++) dp[0][j] = j; // 插入j次
  16.    
  17.     for(int i = 1; i <= m; i++) {
  18.         for(int j = 1; j <= n; j++) {
  19.             if(A[i-1] == B[j-1]) {
  20.                 // 字符相同,无需操作
  21.                 dp[i][j] = dp[i-1][j-1];
  22.             } else {
  23.                 // 取三种操作中的最小值:插入、删除、替换
  24.                 dp[i][j] = min({
  25.                     dp[i][j-1] + 1,    // 插入
  26.                     dp[i-1][j] + 1,    // 删除
  27.                     dp[i-1][j-1] + 1   // 替换
  28.                 });
  29.             }
  30.         }
  31.     }
  32.    
  33.     cout << dp[m][n] << endl;   // 输出最小编辑距离
  34.     return 0;
  35. }
复制代码

五、总结
本文通过动态规划方法解决了洛谷P2758的编辑距离问题,关键点在于理解dp数组的构建逻辑与状态转移方程。通过清晰的边界条件和三种操作的优化选择,实现了高效求解。代码简洁且注释明确,适合算法学习者参考实践。
来源:洛谷题解

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表