动态规划经典题:编辑距离(hard) 详解,看了还不会你来砍我

这篇具有很好参考价值的文章主要介绍了动态规划经典题:编辑距离(hard) 详解,看了还不会你来砍我。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

🧸🧸🧸各位大佬大家好,我是猪皮兄弟🧸🧸🧸
编辑距离 动态规划,数据结构与算法,C++,动态规划,算法


为了更好的理解,我们从易到难的来解决编辑距离的问题

一、最长公共子序列

Leetcode最长公共子序列
编辑距离 动态规划,数据结构与算法,C++,动态规划,算法
一般的,我们在做序列DP问题的时候,遇到两个字符串都会用一个二维数组来进行DP,最长公共子序列简称LCS(longest common subsequence)
对于本题
状态定义:
DP[i][j]表示具有word1[0:i]的字符串和word2[0:j]的字符串中的LCS
比如"abcde" [0:2]也就是abc

状态转移方程:
①当word1[i-1]==word2[j-1]
(因为二维DP中dp[i][0]dp[0][j]代表某个字符串为空,所以要多留一个空间的位置,也就是说dp中的i也就是word中的i-1;或者也可以word1=“”+word1,多一个位置)
那么此时就要从之前的LCS的长度+1
dp[i][j]=dp[i-1][j-1]+1

②当word1[i-1]!=word2[j-1]
此时,就需要找之前的最大公共子序列长度
dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
(其实这里容易想到还有一个dp[i-1][j-1],但是此位置已经被找过,再加上的话就多余了,虽然加上没什么问题)

初始化:
当某个字符串为空的时候,LCS=0;
dp[i][0]和dp[0][j]都是0

int longestCommonSubsequence(string text1, string text2) {
        //dp[i][j] 表示的是当前下的最大公共子序列长度
		//text1[0:i]表示text1的0~i这个长度的子串
        text1=" "+text1;
        text2=" "+text2;
        int m = text1.size(),n=text2.size();
        vector<vector<int>> dp(m+1,vector<int>(n+1,0));
        //dp[0][j]表示空串与text2的子序列,为0  dp[i][0]同理
        for(int i=1;i<m;i++)
        {
            for(int j=1;j<n;j++)
            {
                if(text1[i]==text2[j])//两个相等,则最长子序列+1
                {
                    dp[i+1][j+1] = dp[i][j] +1;
                }
                else //不等则是取其中的最大值,因为dp[i][j]表示到此为止的最大子序列
                {
                    dp[i+1][j+1] = max(dp[i+1][j],dp[i][j+1]);
                }
            }
        }
        
        return dp[m][n];
    }

二、两个字符串的删除操作

其实编辑距离就是该题的延伸,所以这道题其实是很重要的
Leetcode两个字符串的删除操作
编辑距离 动态规划,数据结构与算法,C++,动态规划,算法

思路一:
我们可以转换为求LCS,然后因为这道题只允许删除,所以返回每个字符串的长度和LCS的差值就可以了

int minDistance(string word1, string word2) {
        int m =word1.size();
        int n=word2.size();
        //最长子序列长度
        vector<vector<int>> dp(m+1,vector<int>(n+1,0));
        //dp的i就是word的i-1
        for(int i=1;i<m+1;i++)
        {
            for(int j=1;j<n+1;j++)
            {
                if(word1[i-1]==word2[j-1])
                {
                    dp[i][j]=dp[i-1][j-1]+1;
                }
                else 
                {
                    dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
                }
            }
        }
        return m-dp[m][n]+n-dp[m][n];
    }

思路二:
状态定义:
dp[i][j]表示word1[0:i]和word2[0:j]两个字符串的最少删除数

状态转移方程:
提示:因为二维的DP数组会有多的一个判断空字符串的行和列,所以word中的i-1就是dp中的i
①word1[i-1]==word2[j-1]
那么不需要删除
dp[i][j]=dp[i-1][j-1];

②word1[i-1]!=word2[j-1]
那么就需要删除
dp[i][j]=min(dp[i-1][j],dp[i][j-1])+1;
因为比我少一个字符的字符串的最少删除次数以及确定了,我再+1即可

初始化:
当某个字符串为空时 ,删除次数就等于不为空的字符串长度
dp[i][0]=i;
dp[0][j]=j;

int minDistance(string word1, string word2) {
        int m =word1.size();
        int n=word2.size();
        vector<vector<int>> dp(m+1,vector<int>(n+1,0));
        for(int i=0;i<m+1;i++) dp[i][0]=i;
        for(int j=0;i<n+1;i++) dp[0][j]=j;
        
        for(int i=1;i<m+1;i++)
        {
            for(int j=1;j<n+1;j++)
            {
                if(word1[i-1]==word2[j-1])
                {
                    dp[i][j]=dp[i-1][j-1];
                }
                else 
                {
                    dp[i][j]=min(dp[i-1][j],dp[i][j-1])+1;
                }
            }
        }
        return dp[m][n];
    }

三、编辑距离Hard

Leetcode编辑距离
编辑距离 动态规划,数据结构与算法,C++,动态规划,算法
其实最难的就是dp的定义,dp数组中的值代表什么意思,然后分析好有什么情况,写出状态转移方程基本上就做出来了

状态定义:
dp[i][j]表示word1[0:i]和word2[0:j]这两个字符串的最少编辑次数

状态转移方程:
通过分析,我们发现每个字符串有三种操作 插入/删除/替换
那么,对于两个字符串 A B 就有六种操作
但是,很快发现:
1.对于A的插入其实就相当于对于B的删除
比如A:dog B:doge
2.对于A的删除其实就相当于对于B的插入
比如A:doge B:dog
3.对于A的替换和对于B的替换是等价的

所以我们分析出
①当word1[i-1]==word2[j-1]
那么此时我们不需要进行操作
dp[i][j]=dp[i-1][j-1];
②当word1[i-1]!=word2[j-1]
那么此时就有三种可能
第一种:对A插入/对B删除
第二种:对A删除/对B插入
第三种:对A替换/对B替换
dp[i][j]=max({dp[i][j-1],dp[i-1][j],dp[i-1][j-1]})+1
dp[i-1][j]就表示对A进行插入/对B删除,因为A少一个
同理dp[i][j-1]也是一样
dp[i-1][j-1]表示替换,因为你必须是同时增加一个字符才是替换,不然的话长度不对,只能是在上一个的基础上插入或者删除

初始化:
当有一个串为空串,那么
dp[i][0]=i ;
dp[0][j]=j ;

int minDistance(string word1, string word2) {
        //序列dp的套路,两个字符串一般用二维dp
        int m = word1.size();
        int n=word2.size();
        vector<vector<int>> dp(m+1,vector<int>(n+1,0));//多开一个的原因是dp[i][0]和dp[0][j]有一个串是空串
        for(int i=0;i<m+1;i++) 
            dp[i][0]=i;
        for(int j=0;j<n+1;j++) 
            dp[0][j]=j;
        //初始化两个串某一个为空的操作
        for(int i=1;i<m+1;i++)
        {
            for(int j=1;j<n+1;j++)
            {
                if(word1[i-1]==word2[j-1]) dp[i][j]=dp[i-1][j-1];//多一个字符串的话肯定要做修改
                else
                {
                    dp[i][j]=min({dp[i-1][j-1],dp[i-1][j],dp[i][j-1]})+1;
                }
            }
        }
        return dp[m][n];
    }

编辑距离 动态规划,数据结构与算法,C++,动态规划,算法文章来源地址https://www.toymoban.com/news/detail-814939.html

到了这里,关于动态规划经典题:编辑距离(hard) 详解,看了还不会你来砍我的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【动态规划】求解编辑距离问题

    编辑距离问题是求解将⼀个字符串转换为另⼀个字符串所需的 插⼊、删除、替换 的最小次数。 C O M M O M → s u b C O M M U M → s u b C O M M U N → i n s C O M M U N E mathbb{COMMOM} overset{sub}{rightarrow} mathbb{COMMUM} overset{sub}{rightarrow}mathbb{COMMUN} overset{ins}{rightarrow} mathbb{COMMUNE} COMMOM →

    2024年02月02日
    浏览(37)
  • 【Leetcode72】编辑距离(动态规划)

    (1)确定状态 dp[i][j] 表示word1中前 i 个单词,变换到word2中前 j 个字符,最少需要的动作次数。 (2)状态转移方程 编辑距离可以通过以下三种操作进行计算:插入一个字符、删除一个字符、替换一个字符。因此,可以使用以下状态转移方程: 如果 A[i] == B[j] ,那么 dp[i][j]

    2024年02月12日
    浏览(31)
  • 【算法】动态规划算法求(编辑距离)

    目录 编辑距离: 举例: 代码如下 调试: 核心代码: 画图演示上述代码:    是一种计算两个自符串之间差异程度的方法,它通过比较 两个字符串之间的插入,删除和 替换操作的数量 ,来确定他们之间的距离。 现有两个字符串 字符串s1:”CTGA\\\" 字符串s2:  \\\"ACGCTA\\\" 求s1和

    2024年02月12日
    浏览(30)
  • 【动态规划----“最小编辑距离”问题(C++解决)】

    给定两个字符串A和B,以及下列三种字符运算: (1)删除一个字符(2)插入一个字符(3)将一个字符改写为另一个字符 设计算法求将A通过以上三种操作转换为B的最小次数 举例: “xy” = “xz”,只需要把 y 替换成 z,因此,最小编辑距离为 1。 “xyz” = “xy”,只需要删除

    2024年04月25日
    浏览(75)
  • 动态规划Day16(编辑距离,删除元素待写完)

    目录 583. 两个字符串的删除操作 看到题目的第一想法                看到代码随想录之后的想法 自己实现过程中遇到的困难(看代码) 72. 编辑距离 看到题目的第一想法                看到代码随想录之后的想法 自己实现过程中遇到的困难(看代码) 力扣题目链接(opens new

    2024年01月24日
    浏览(41)
  • 动态规划问题-最小编辑距离(Minimum Edit Distance)

    我们今天要探讨的动态规划问题来源于俄罗斯科学家Levenshtein提出的两个对象之间的不相似度,在音频、语言翻译等领域有广泛的应用。如果用于评估字符串之间的不相似度,那么又称为最小编辑距离MED(Minimum Edit Distance),它规定从string 1到转换成 string 2的最少操作数,最少操

    2024年02月09日
    浏览(44)
  • 【LeetCode:72. 编辑距离 | 暴力递归=>记忆化搜索=>动态规划 】

    🚀 算法题 🚀 🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀 🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨ 🌲 作者简介:硕风和炜,CSDN-Java领域新星创作者🏆,保研|国家奖学金|高中学习JAVA|大学完善JAVA开发技术栈|面试刷题|面经八股文

    2024年02月06日
    浏览(31)
  • 动态规划16 | ● 583. 两个字符串的删除操作 ● *72. 编辑距离

    https://programmercarl.com/0583.%E4%B8%A4%E4%B8%AA%E5%AD%97%E7%AC%A6%E4%B8%B2%E7%9A%84%E5%88%A0%E9%99%A4%E6%93%8D%E4%BD%9C.html 考点 子序列问题 我的思路 dp[i][j]的含义是,当两个字符串分别取前i和j个元素时,对应的最少相等删除步数是多少 递推公式为,如果两个字符串第i和j个元素恰好相等,则dp值应

    2024年04月22日
    浏览(35)
  • LeetCode | C++ 动态规划——583. 两个字符串的删除操作、72. 编辑距离

    583题目链接 做法一: 本题和1143.最长公共子序列基本相同,只要求出两个字符串的最长公共子序列长度即可,那么除了最长公共子序列之外的字符都是必须删除的,最后用两个字符串的总长度减去两个最长公共子序列的长度就是删除的最少步数。 做法二: 本题和115.不同的子

    2024年02月16日
    浏览(41)
  • 动态规划详细讲解c++|经典例题讲解认识动态规划|0-1背包问题详解

    uu们,你们好!这次的分享是动态规划,其中介绍了动态规划的相关概念和做题模板(三要素),同时为了uu们对动态规划方法有更加形象的认识,特地找了两个经典问题,和大家一起分析。并且呢,为了大家检验自己的学习成果,我专门在常用的oj上为uu们找到了相关题目的

    2024年04月11日
    浏览(49)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包