研习代码 day48 | 动态规划——终极子序列问题(编辑距离)

2023-12-06 04:36

本文主要是介绍研习代码 day48 | 动态规划——终极子序列问题(编辑距离),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

        1.1 题目

        给定两个单词 word1 和 word2 ,返回使得 word1 和  word2 相同所需的最小步数

        每步 可以删除任意一个字符串中的一个字符。

示例 1:

输入: word1 = "sea", word2 = "eat"
输出: 2
解释: 第一步将 "sea" 变为 "ea" ,第二步将 "eat "变为 "ea"

示例  2:

输入:word1 = "leetcode", word2 = "etco"
输出:4

提示:

  • 1 <= word1.length, word2.length <= 500
  • word1 和 word2 只包含小写英文字母

        1.2 题目链接

        583.两个字符串的删除操作

        1.3 解题过程和过程想法

        (1)解题过程

        通过最长公共子序列求最少的删除操作:先求出最长公共子序列,再用两串的总长度-2*最长公共子序列长度,即得到最少需删除操作的次数
        分析:当前的匹配情况会受到之前元素的情况所影响,且影响的方式是类似的,考虑采用动态规划的策略。
        # 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最长公共子序列长度为dp[i][j]
        # 递推关系:若二者元素相匹配,当前情况取决于 用或不用 当前的元素,
                                  dp[i][j] = dp[i-1][j-1] + dp[i-1][j]
                             若二者元素不匹配,当前情况的结果与不用当前元素的情况相同
                                  dp[i][j] = dp[i-1][j]

图片来源:代码随想录,红色文字是自己加的


        # 初始化:由上述递推关系可知当前位置的填写是基于左上方和正上方的元素,所以需要提前对首行首列进行初始赋值
                                dp[0][j] = 0         # 首行:没有母串,直接赋值 0
                                dp[i][0] = 1         # 首列:没有子串,即空子串,赋值1

        直接迭代当前最少的删除操作:
        
当前的匹配情况会受到之前元素的情况所影响,且影响的方式是类似的,考虑采用动态规划的策略。
        # 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最少需删除的长度为dp[i][j]
                dp = [[0]*(n+1) for _ in range(m+1)]

        # 递推关系:若两指针所指元素相同,更新当前数组值不需删除,即不更新 dp[i][j] = dp[i-1][j-1]
                             否则更新当前位置 dp[i][j] = min(dp[i-1][j]+1,dp[i][j-1],dp[i-1][j-1]+2)
注:不等时有三种情况——删第一个串中的元素,删第二个串中的元素,同时删除两个串中的元素
        # 初始化:因为当前位置的值由左上、正上方、左方推导,所以初始化首行首列
                            dp[0][j] = j    # 其中一个是空串,另一个串长度为 j 时,需删 j 个位置
                            dp[i][0] = i    # 其中一个是空串,另一个串长度为 i 时,需删 i 个位置

        (2)过程想法

        由于第一次做此类题目,第一种解法最先想到,后者是现学的

        1.4 代码

        1.4.1 通过最长公共子序列求最少的删除操作
class Solution:def minDistance(self, word1: str, word2: str) -> int:m = len(word1)n = len(word2)# 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最长公共子序列长度为dp[i][j]dp = [[0]*(n+1) for _ in range(m+1)]# 递推关系:因为判断的不一定是连续的情况,直接迭代,dp[i][j] = dp[i-1][j-1] + 1for i in range(1,m+1):for j in range(1,n+1):if word1[i-1] == word2[j-1]:dp[i][j] = dp[i-1][j-1] + 1else:dp[i][j] = max(dp[i-1][j], dp[i][j-1])return m+n-2*dp[m][n]
        1.4.2 直接迭代当前最少的删除操作
class Solution:def minDistance(self, word1: str, word2: str) -> int:m = len(word1)n = len(word2)# 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最少需删除的长度为dp[i][j]dp = [[0]*(n+1) for _ in range(m+1)]# 递推关系:若两指针所指元素相同,更新当前数组值不需要删除,即不更新 dp[i][j] = dp[i-1][j-1];# 否则更新当前位置 dp[i][j] = min(dp[i-1][j]+1,dp[i][j-1],dp[i-1][j-1]+2)# 初始化:因为当前位置的值由左上、正上方、左方推导,所以初始化首行首列for j in range(n+1):dp[0][j] = j    # 其中一个是空串,另一个串长度为 j 时,需删 j 个位置for i in range(m+1):dp[i][0] = i    # 其中一个是空串,另一个串长度为 i 时,需删 i 个位置for i in range(1,m+1):for j in range(1,n+1):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][j-1]+1,dp[i-1][j-1]+2)return dp[m][n]

二、编辑距离

        2.1 题目

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

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

  • 插入一个字符
  • 删除一个字符
  • 替换一个字符

示例 1:

输入:word1 = "horse", word2 = "ros"
输出:3
解释:
horse -> rorse (将 'h' 替换为 'r')
rorse -> rose (删除 'r')
rose -> ros (删除 'e')

示例 2:

输入:word1 = "intention", word2 = "execution"
输出:5
解释:
intention -> inention (删除 't')
inention -> enention (将 'i' 替换为 'e')
enention -> exention (将 'n' 替换为 'x')
exention -> exection (将 'n' 替换为 'c')
exection -> execution (插入 'u')

提示:

  • 0 <= word1.length, word2.length <= 500
  • word1 和 word2 由小写英文字母组成

        2.2 题目链接

        72.编辑距离

        2.3 解题过程和过程想法

        (1)解题过程        

        分析:当前的匹配情况会受到之前元素的情况所影响,且影响的方式是类似的,考虑采用动态规划的策略。
        # 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最少需操作的次数为dp[i][j]
                dp = [[0]*(n+1) for _ in range(m+1)]

        # 递推关系:若两指针所指元素相同,更新当前数组值不需操作,即不更新 dp[i][j] = dp[i-1][j-1]
                             否则更新当前位置 dp[i][j] = min(dp[i-1][j]+1, dp[i][j-1]+1, dp[i-1][j-1]+1)
               注:不等时有三种操作:删长串中的元素,增加短串中的元素,替换一个串中的元素
        # 初始化:因为当前位置的值由左上、正上方、左方推导,所以初始化首行首列
                            dp[0][j] = j    # 其中一个是空串,另一个串长度为 j 时,需操作 j 个位置
                            dp[i][0] = i    # 其中一个是空串,另一个串长度为 i 时,需操作 i 个位置

        (2)过程想法

        解题思路与上一题类似,只是可操作的细节略有不同

        2.4 代码

class Solution:def minDistance(self, word1: str, word2: str) -> int:m = len(word1)n = len(word2)# 数组:以i-1为结尾的word1字符串与以j-1为结尾的word2中最少需操作的次数为dp[i][j]dp = [[0]*(n+1) for _ in range(m+1)]# 递推关系:若两指针所指元素相同,更新当前数组值不需要操作,即不更新 dp[i][j] = dp[i-1][j-1];# 否则更新当前位置 dp[i][j] = min(dp[i-1][j]+1,dp[i][j-1]+1,dp[i-1][j-1]+1)# 初始化:因为当前位置的值由左上、正上方、左方推导,所以初始化首行首列for j in range(n+1):dp[0][j] = j    # 其中一个是空串,另一个串长度为 j 时,需操作 j 个位置for i in range(m+1):dp[i][0] = i    # 其中一个是空串,另一个串长度为 i 时,需操作 i 个位置for i in range(1,m+1):for j in range(1,n+1):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][j-1]+1,dp[i-1][j-1]+1)return dp[m][n]

这篇关于研习代码 day48 | 动态规划——终极子序列问题(编辑距离)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/460461

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

动态规划---打家劫舍

题目: 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。 思路: 动态规划五部曲: 1.确定dp数组及含义 dp数组是一维数组,dp[i]代表

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

软考系统规划与管理师考试证书含金量高吗?

2024年软考系统规划与管理师考试报名时间节点: 报名时间:2024年上半年软考将于3月中旬陆续开始报名 考试时间:上半年5月25日到28日,下半年11月9日到12日 分数线:所有科目成绩均须达到45分以上(包括45分)方可通过考试 成绩查询:可在“中国计算机技术职业资格网”上查询软考成绩 出成绩时间:预计在11月左右 证书领取时间:一般在考试成绩公布后3~4个月,各地领取时间有所不同

uva 10131 最长子序列

题意: 给大象的体重和智商,求体重按从大到小,智商从高到低的最长子序列,并输出路径。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vect