LeetCode算法题解(动态规划)|LeetCode583. 两个字符串的删除操作、LeetCode72. 编辑距离

本文主要是介绍LeetCode算法题解(动态规划)|LeetCode583. 两个字符串的删除操作、LeetCode72. 编辑距离,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

题目链接:583. 两个字符串的删除操作
题目描述:

给定两个单词 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 只包含小写英文字母
算法分析:
定义dp数组及下标含义:

dp[i][j]表示以下标i结尾的word1字符串和以下标j结尾的word2字符串,删除至相同所需的最小步骤。

递推公式:

如果word1[i]==word2[j],无需进行删除操作,那么dp[i][j] == dp[i-1][j-1];

如果word1[i] !=word2[j],那么dp[i][j]可以由三个方向推导出来:

1、删除word1[i],删除word2[j],两个步骤,即dp[i][j]=dp[i-1][j-1]+2;

2、删除word1[i],word2[j]继续匹配,一个步骤,即dp[i][j]=dp[i-1][j]+1;

3、删除word2[j],word1[i]继续匹配,一个步骤,即dp[i][j]=dp[i][j-1]+1;

所以dp[i][j]=max(dp[i-1][j-1]+2,dp[i-1][j]+1,dp[i][j-1]+1);

初始化:

由递推公式可知,dp需要初始化第一行和第一列。

        for(int j = 0; j < word2.length(); j++){if(word1.charAt(0) == word2.charAt(j)){//如果相等,那么word1[0]和以下标j结尾的word2字符串相同所需的操作步骤为j,即删掉除word2[j]之前的所有字符。dp[0][j] = j;}else{//如果不相等,当j == 0 的时候,需要删除word1[0]和word2[0],2个步骤//当j!=0时,在之前步骤的基础上,删除word2[j],加一个步骤。if(j > 0) dp[0][j] = dp[0][j-1] + 1;else dp[0][j] = 2;}}for(int i = 0; i < word1.length(); i++){//第一列同理if(word1.charAt(i) == word2.charAt(0)){dp[i][0] = i;}else{if(i > 0) dp[i][0] = dp[i-1][0] + 1;else dp[i][0] = 2;}}
遍历顺序:

word1和word2的哪个先遍历都无所谓,只要都是从前往后遍历。

打印dp数组进行验证。

代码如下:

class Solution {public int minDistance(String word1, String word2) {int[][] dp = new int[word1.length()][word2.length()];for(int j = 0; j < word2.length(); j++){if(word1.charAt(0) == word2.charAt(j)){//如果相等,那么word1[0]和以下标j结尾的word2字符串相同所需的操作步骤为j,即删掉除word2[j]之前的所有字符。dp[0][j] = j;}else{//如果不相等,当j == 0 的时候,需要删除word1[0]和word2[0],2个步骤//当j!=0时,在之前步骤的基础上,删除word2[j],加一个步骤。if(j > 0) dp[0][j] = dp[0][j-1] + 1;else dp[0][j] = 2;}}for(int i = 0; i < word1.length(); i++){//第一列同理if(word1.charAt(i) == word2.charAt(0)){dp[i][0] = i;}else{if(i > 0) dp[i][0] = dp[i-1][0] + 1;else dp[i][0] = 2;}}for(int i = 1; i < word1.length(); i++) {for(int j = 1; j < word2.length(); j++) {if(word1.charAt(i) == word2.charAt(j)){//相等时由左上方推出来dp[i][j] = dp[i-1][j-1];}else{//不相等时由左方、上方、左上方推出来dp[i][j] = Math.min(dp[i-1][j-1] + 2,Math.min(dp[i-1][j] + 1,dp[i][j-1] + 1));}}}return dp[word1.length()-1][word2.length()-1];}
}

二、LeetCode72. 编辑距离

题目链接:72. 编辑距离
题目描述:

给你两个单词 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 由小写英文字母组成
算法分析:
定义dp数组及下标含义:

dp[i][j]表示以下标i结尾的字符串word1转换成以下标j结束的字符串word2所需的最少操作步骤。

递推公式:

当word1[i]==word2[j]时,无需任何操作,即dp[i][j]=dp[i-1][j-1];

当word1[i]!=word2[j]时,可以有三种操作:

1、替换word1[i]元素,使之与word2[j]相等,即dp[i][j]=dp[i-1][j-1]+1;

2、删除word1[i]元素,即dp[i][j]=dp[i-1][j] + 1;

3、在word1[i]元素之后插入一个元素,使该元素与word2[j]相等,那么即dp[i][j]=dp[i][j-1]+1;

初始话:

根据递推公式初始化dp数组的第一行和第一列。

        boolean flag = false;for(int j = 0; j < word2.length(); j++) {if(word1.charAt(0) == word2.charAt(j)){//如果相等,那么word1[0]转化成以j结尾的word2字符串所需操作数为j,即添加j个数,注意j是从0开始的。dp[0][j] = j;flag = true;}else{//如果不相等,当word2[j]之前有出现过word1[0]时,只需操作j步,即在word1[0]添加j个数//当word2[j]之前没有出现果word1[0]时,需要操作j+1步,即将word1[0]转化成word1[0],再添加j个数if(flag) dp[0][j] = j;else dp[0][j] = j + 1;}}flag = false;for(int i = 0; i < word1.length(); i++) {//初始化第一列同理if(word1.charAt(i) == word2.charAt(0)){dp[i][0] = i;flag = true;}else{if(flag) dp[i][0] = i;else dp[i][0] = i + 1;}}
遍历顺序:

word1和word2哪个先遍历没关系,但都需从前往后遍历。

打印dp数组进行验证。

代码如下:

class Solution {public int minDistance(String word1, String word2) {if(word1.length() == 0) return word2.length();if(word2.length() == 0) return word1.length();int[][] dp = new int[word1.length()][word2.length()];boolean flag = false;for(int j = 0; j < word2.length(); j++) {if(word1.charAt(0) == word2.charAt(j)){//如果相等,那么word1[0]转化成以j结尾的word2字符串所需操作数为j,即添加j个数,注意j是从0开始的。dp[0][j] = j;flag = true;}else{//如果不相等,当word2[j]之前有出现过word1[0]时,只需操作j步,即在word1[0]添加j个数//当word2[j]之前没有出现果word1[0]时,需要操作j+1步,即将word1[0]转化成word1[0],再添加j个数if(flag) dp[0][j] = j;else dp[0][j] = j + 1;}}flag = false;for(int i = 0; i < word1.length(); i++) {//初始化第一列同理if(word1.charAt(i) == word2.charAt(0)){dp[i][0] = i;flag = true;}else{if(flag) dp[i][0] = i;else dp[i][0] = i + 1;}}for(int i = 1; i < word1.length(); i++) {for(int j = 1; j < word2.length(); j++) {if(word1.charAt(i) == word2.charAt(j)) {//如果相等,无需操作dp[i][j] = dp[i-1][j-1];}else{//如果不相等,再三个操作当中取步骤最少的操作数。dp[i][j] = Math.min(dp[i-1][j-1] + 1, Math.min(dp[i-1][j] + 1, dp[i][j-1] + 1));}}}// for(int i = 0; i < word1.length(); i++) {//     for(int j = 0; j < word2.length(); j++) {//         System.out.print(dp[i][j] + " ");//     }//     System.out.println();// }return dp[word1.length()-1][word2.length()-1];}
}

总结

这两道题比较类似,只要会其中一道题,两外一个也很容易想出来。

这篇关于LeetCode算法题解(动态规划)|LeetCode583. 两个字符串的删除操作、LeetCode72. 编辑距离的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Python如何计算两个不同类型列表的相似度

《Python如何计算两个不同类型列表的相似度》在编程中,经常需要比较两个列表的相似度,尤其是当这两个列表包含不同类型的元素时,下面小编就来讲讲如何使用Python计算两个不同类型列表的相似度吧... 目录摘要引言数字类型相似度欧几里得距离曼哈顿距离字符串类型相似度Levenshtein距离Jaccard相

0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型的操作流程

《0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeekR1模型的操作流程》DeepSeekR1模型凭借其强大的自然语言处理能力,在未来具有广阔的应用前景,有望在多个领域发... 目录0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型,3步搞定一个应

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

docker如何删除悬空镜像

《docker如何删除悬空镜像》文章介绍了如何使用Docker命令删除悬空镜像,以提高服务器空间利用率,通过使用dockerimage命令结合filter和awk工具,可以过滤出没有Tag的镜像,并将... 目录docChina编程ker删除悬空镜像前言悬空镜像docker官方提供的方式自定义方式总结docker

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多