【DP】编辑距离

日常吐槽:关于DP,有一种莫名的恐惧…maybe源于与mtw大佬与quantum11大佬,初中时抬老师爬楼梯的经历。。。

言归正传:

编辑距离

【题目描述】
设A和B是两个字符串。我们要用最少的字符操作次数,将字符串A转换为字符串B。这里所说的字符操作共有三种:

1、删除一个字符;

2、插入一个字符;

3、将一个字符改为另一个字符。

对任意的两个字符串A和B,计算出将字符串A变换为字符串B所用的最少字符操作次数。

【输入】
第一行为字符串A;第二行为字符串B;字符串A和B的长度均小于2000。

【输出】
只有一个正整数,为最少字符操作次数。

【输入样例】
sfdqxbw
gfdgw
【输出样例】
4

【思路】

比较基础的一道DP题目,(我才不会告诉你们,我也是看了书才会做)~
既然是DP,那么我们分析一下子问题,当前处理A到第i个字符,处理B到第j个字符
状态f[i][j]代表此时最小的编辑距离
不难推出:

F[I][J]=MIN{F[I-1][J-1],F[I-1][J],F[I][J-1]}+1;

好简单

的样子
额…

不存在的

如果A[I]==B[I]的话,我们发现

根本不用修改

then:

if(a[i]==b[j]) 
    f[i][j]=min(min(f[i-1][j]+1,f[i][j-1]+1),f[i-1][j-1]);
else
    f[i][j]=min(min(f[i-1][j],f[i][j-1]),f[i-1][j-1])+1;

大概就是这个样子的

吧…

接下来

关于边界处理

假设A空:则 for(int j=0;j<=m;j++) f[0][j]=j;
假设B空:则 for(int i=0;i<=n;i++) f[i][0]=i;

就这样…完…完成了?

不存在的

才怪呢…

附上代码

#include<iostream>
#include<cstdio>
#include<cctype>
#include<cstring> 
#include<algorithm>
using namespace std;

inline int read()
{
    char chr=getchar();
    int f=1,ans=0;
    while(!isdigit(chr)) {if(chr=='-') f=-1;chr=getchar();}
    while(isdigit(chr))  {ans=ans*10;ans+=chr-'0';chr=getchar();}
    return ans*f;

}
int f[2005][2005];
int n,m;
char a[2005],b[2005];
int main()
{
    scanf("%s\n%s",a+1,b+1);
    n=strlen(a+1);
    m=strlen(b+1);
    for(int i=0;i<=n;i++)   f[i][0]=i;
    for(int j=0;j<=m;j++)   f[0][j]=j;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
        {
            if(a[i]==b[j]) 
                f[i][j]=min(min(f[i-1][j]+1,f[i][j-1]+1),f[i-1][j-1]);
            else
                f[i][j]=min(min(f[i-1][j],f[i][j-1]),f[i-1][j-1])+1;
        }
    cout<<f[n][m];
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值