【算法与数据结构】583、72、LeetCode两个字符串的删除操作+编辑距离

文章目录

  • 一、583、两个字符串的删除操作
  • 二、72、编辑距离
  • 三、完整代码

所有的LeetCode题解索引,可以看这篇文章——【算法和数据结构】LeetCode题解。

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

在这里插入图片描述

  思路分析:本题的思路和115、不同的子序列差不多,只是变成了两个字符串都能删除字符。

  1. 第一步,动态数组的含义。 d p [ i ] [ j ] dp[i][j] dp[i][j]代表使得 w o r d 1 [ 0 , i − 1 ] word1[0, i-1] word1[0,i1] w r o d 2 [ 0 , j − 1 ] wrod2[0, j-1] wrod2[0,j1]相等所需删除的最小步数。
  2. 第二步,递推公式。 d p [ i ] [ j ] dp[i][j] dp[i][j]可以由两种情况推导出来:
  • w o r d 1 [ i − 1 ] word1[i - 1] word1[i1] w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]相同:那么最小步数和使得 w o r d 1 [ 0 , i − 2 ] word1[0, i-2] word1[0,i2] w r o d 2 [ 0 , j − 2 ] wrod2[0, j-2] wrod2[0,j2]相等所需删除的最小步数相同。 d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] dp[i][j] = dp[i-1][j-1] dp[i][j]=dp[i1][j1]
  • w o r d 1 [ i − 1 ] word1[i - 1] word1[i1] w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]不相同:这种情况的 d p [ i ] [ j ] dp[i][j] dp[i][j]可以由三部分构成:若 w o r d 1 [ 0 , i − 2 ] word1[0, i - 2] word1[0,i2] w o r d 2 [ 0 , j − 1 ] word2[0, j - 1] word2[0,j1]做了 d p [ i − 1 ] [ j ] dp[i - 1][j] dp[i1][j]次删除操作以后会相等,那么再删除 w o r d 1 [ i − 1 ] word1[i - 1] word1[i1]以后又可以相等,即 d p [ i ] [ j ] = d p [ i − 1 ] [ j ] + 1 dp[i][j] = dp[i - 1][j] + 1 dp[i][j]=dp[i1][j]+1;若 w o r d 1 [ 0 , i − 1 ] word1[0, i - 1] word1[0,i1] w o r d 2 [ 0 , j − 2 ] word2[0, j - 2] word2[0,j2]做了 d p [ i ] [ j − 1 ] dp[i][j - 1] dp[i][j1]次删除操作以后会相等,那么再删除 w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]以后又可以相等,即 d p [ i ] [ j ] = d p [ i ] [ j − 1 ] + 1 dp[i][j] = dp[i][j - 1] + 1 dp[i][j]=dp[i][j1]+1。因为要求最小步数,那么我们对两项取最小: d p [ i ] [ j ] = m i n ( d p [ i − 1 ] [ j ] + 1 , d p [ i ] [ j − 1 ] + 1 ) dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1) dp[i][j]=min(dp[i1][j]+1,dp[i][j1]+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], dp[i][j - 1]) + 1;
  1. 第三步,元素初始化。 d p [ i ] [ 0 ] dp[i][0] dp[i][0](第一列)表示字符串 w o r d 1 [ 0 , i − 1 ] word1[0, i-1] word1[0,i1]变成空字符串需要删除的最小字符个数。 d p [ 0 ] [ j ] dp[0][j] dp[0][j](第一行)表示 w o r d 2 [ 0 , j − 1 ] word2[0, j-1] word2[0,j1]变成空字符串需要删除的最小字符个数。其中,空字符串word1变成空字符串word2的个数为0。那么 d p [ 0 ] [ 0 ] = 0 , d p [ i ] [ 0 ] = i , d p [ 0 ] [ j ] = j dp[0][0]=0, dp[i][0] = i, dp[0][j] = j dp[0][0]=0,dp[i][0]=i,dp[0][j]=j
		for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为i
  1. 第四步,递归顺序。一共有两层循环,从前往后进行遍历。
  2. 第五步,打印结果。

在这里插入图片描述

  程序如下

// 583、两个字符串的删除操作-动态规划
class Solution {
public:int minDistance(string word1, string word2) {vector<vector<uint64_t>> dp(word1.size() + 1, vector<uint64_t>(word2.size() + 1, 0));	// dp[0][0]为0for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为ifor (int i = 1; i <= word1.size(); i++) {for (int j = 1; j <= word2.size(); 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[word1.size()][word2.size()];}
};

复杂度分析:

  • 时间复杂度: O ( n ∗ m ) O(n*m) O(nm) n n n m m m分别是两个字符串的长度。
  • 空间复杂度: O ( n ∗ m ) O(n*m) O(nm)

二、72、编辑距离

在这里插入图片描述

  思路分析:本题在583题的基础之上加入了插入和替换操作。我们同样用动态规划的方法分析。

  1. 第一步,动态数组的含义。 d p [ i ] [ j ] dp[i][j] dp[i][j]代表使得 w o r d 1 [ 0 , i − 1 ] word1[0, i-1] word1[0,i1] w r o d 2 [ 0 , j − 1 ] wrod2[0, j-1] wrod2[0,j1]相等所需的最小操作数。
  2. 第二步,递推公式。 d p [ i ] [ j ] dp[i][j] dp[i][j]可以由两种情况推导出来:
  • w o r d 1 [ i − 1 ] word1[i - 1] word1[i1] w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]相同:那么最小操作数和使得 w o r d 1 [ 0 , i − 2 ] word1[0, i-2] word1[0,i2] w r o d 2 [ 0 , j − 2 ] wrod2[0, j-2] wrod2[0,j2]相等所需的最小操作数相同。 d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] dp[i][j] = dp[i-1][j-1] dp[i][j]=dp[i1][j1]
  • w o r d 1 [ i − 1 ] word1[i - 1] word1[i1] w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]不相同:这种情况的 d p [ i ] [ j ] dp[i][j] dp[i][j]可以由三部分构成:增加、删除和替换。删除部分和583题一致: d p [ i ] [ j ] = m i n ( d p [ i − 1 ] [ j ] + 1 , d p [ i ] [ j − 1 ] + 1 ) dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1) dp[i][j]=min(dp[i1][j]+1,dp[i][j1]+1)。而增加字符和删除的操作数没有区别。若 w o r d 1 [ 0 , i − 2 ] word1[0, i - 2] word1[0,i2] w o r d 2 [ 0 , j − 2 ] word2[0, j - 2] word2[0,j2]做了 d p [ i − 1 ] [ j − 1 ] dp[i - 1][j - 1] dp[i1][j1]次删除操作以后会相等,那么再替换 w o r d 1 [ i − 1 ] word1[i - 1] word1[i1]或者 w o r d 2 [ j − 1 ] word2[j - 1] word2[j1]之间任意一个元素以后又可以相等,即 d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] + 1 dp[i][j] = dp[i - 1][j - 1] + 1 dp[i][j]=dp[i1][j1]+1。因为要求最小操作数,那么我们对两项取最小: d p [ i ] [ j ] = m i n ( d p [ i − 1 ] [ j ] + 1 , d p [ i ] [ j − 1 ] + 1 , d p [ i − 1 ] [ j − 1 ] + 1 ) = m i n ( d p [ i − 1 ] [ j ] , d p [ i ] [ j − 1 ] , d p [ i − 1 ] [ j − 1 ] ) + 1 dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1, dp[i - 1][j - 1] + 1) = min(dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1]) + 1 dp[i][j]=min(dp[i1][j]+1,dp[i][j1]+1,dp[i1][j1]+1)=min(dp[i1][j],dp[i][j1],dp[i1][j1])+1
	if (word1[i - 1] == word2[j - 1]) dp[i][j] = dp[i - 1][j - 1];//else dp[i][j] = min(min(dp[i - 1][j], dp[i][j - 1]), dp[i - 1][j - 1]) + 1;	// min()函数只接受两个参数,或者数组else dp[i][j] = min({ dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1] }) + 1;
  1. 第三步,元素初始化。 d p [ i ] [ 0 ] dp[i][0] dp[i][0](第一列)表示字符串 w o r d 1 [ 0 , i − 1 ] word1[0, i-1] word1[0,i1]变成空字符串需要的最少操作数。 d p [ 0 ] [ j ] dp[0][j] dp[0][j](第一行)表示 w o r d 2 [ 0 , j − 1 ] word2[0, j-1] word2[0,j1]变成空字符串需要的最少操作数。其中,空字符串word1变成空字符串word2的个数为0。那么 d p [ 0 ] [ 0 ] = 0 , d p [ i ] [ 0 ] = i , d p [ 0 ] [ j ] = j dp[0][0]=0, dp[i][0] = i, dp[0][j] = j dp[0][0]=0,dp[i][0]=i,dp[0][j]=j
	for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为i
  1. 第四步,递归顺序。一共有两层循环,从前往后进行遍历。
  2. 第五步,打印结果。

在这里插入图片描述

  程序如下

// 72、编辑距离-动态规划
class Solution2 {
public:int minDistance(string word1, string word2) {vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1, 0));	// dp[0][0]为0for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为ifor (int i = 1; i <= word1.size(); i++) {for (int j = 1; j <= word2.size(); j++) {if (word1[i - 1] == word2[j - 1]) dp[i][j] = dp[i - 1][j - 1];//else dp[i][j] = min(min(dp[i - 1][j], dp[i][j - 1]), dp[i - 1][j - 1]) + 1;	// min()函数只接受两个参数,或者数组else dp[i][j] = min({ dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1] }) + 1;}}return dp[word1.size()][word2.size()];}
};

复杂度分析:

  • 时间复杂度: O ( n ∗ m ) O(n*m) O(nm) n n n m m m分别是两个字符串的长度。
  • 空间复杂度: O ( n ∗ m ) O(n*m) O(nm)

三、完整代码

# include <iostream>
# include <vector>
# include <string>
using namespace std;// 583、两个字符串的删除操作-动态规划
class Solution {
public:int minDistance(string word1, string word2) {vector<vector<uint64_t>> dp(word1.size() + 1, vector<uint64_t>(word2.size() + 1, 0));	// dp[0][0]为0for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为ifor (int i = 1; i <= word1.size(); i++) {for (int j = 1; j <= word2.size(); 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[word1.size()][word2.size()];}
};// 72、编辑距离-动态规划
class Solution2 {
public:int minDistance(string word1, string word2) {vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1, 0));	// dp[0][0]为0for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;		// 第一列初始化为ifor (int j = 1; j <= word2.size(); j++) dp[0][j] = j;		// 第一行初始化为ifor (int i = 1; i <= word1.size(); i++) {for (int j = 1; j <= word2.size(); j++) {if (word1[i - 1] == word2[j - 1]) dp[i][j] = dp[i - 1][j - 1];//else dp[i][j] = min(min(dp[i - 1][j], dp[i][j - 1]), dp[i - 1][j - 1]) + 1;	// min()函数只接受两个参数,或者数组else dp[i][j] = min({ dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1] }) + 1;}}return dp[word1.size()][word2.size()];}
};int main() {//string word1 = "sea", word2 = "eat";	// 测试案例//Solution s1;//int result = s1.minDistance(word1, word2);string word1 = "horse", word2 = "ros";	// 测试案例Solution2 s1;int result = s1.minDistance(word1, word2);cout << result << endl;system("pause");return 0;
}

end

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/452993.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#获取当前CPU,内存使用详情及常用WMI查询语句

1&#xff0c;获取当前CPU使用率&#xff08;需添加System.Management.dll&#xff09;&#xff1a; string queryStr "select * from win32_processor";System.Management.ManagementObjectSearcher searcher new System.Management.ManagementObjectSearcher(qu…

【MATLAB源码-第132期】基于matlab的淘金优化算法(GRO)无人机三维路径规划,输出做短路径图和适应度曲线

操作环境&#xff1a; MATLAB 2022a 1、算法描述 淘金优化算法&#xff08;GoldRush Optimizer&#xff0c;简称GRO&#xff09;是一种启发式优化算法&#xff0c;它受到淘金过程的启发。在淘金过程中&#xff0c;淘金者在河流或矿区中寻找金矿&#xff0c;通过筛选沙砾来寻…

数据库学习笔记2024/2/4

随笔 1. 为什么学? 认识数据,熟悉数据,掌握数据。 进企业必备技能。 2. 怎么学? 1、MySQL数据库就是存储和管理数据的一个大型软件,这个软件有一个专门的语言叫SQL,主要学的是SQL语言,但想要达到企业用人标准,就还得学会熟练使用MySQL这个软件。 2、学习分三阶段: 一. …

mysql:事务的特性ACID、并发事务(脏读、不可重复读、幻读、如何解决、隔离级别)、undo log和redo log的区别、相关面试题和答案

事务是一组操作的集合&#xff0c;它会把所有的操作作为一个整体一起向系统提交或撤销操作请求&#xff0c;即这些操作要么同时成功&#xff0c;要么同时失败。 事务的特性&#xff08;ACID&#xff09; 原子性&#xff08;Atomicity&#xff09;&#xff1a;事务是不可分割的…

Angular BaseView抽离页面公用属性

前言 如果有一系列的页面布局很类似&#xff0c;为了节省时间&#xff0c;我们可以把这些类似的页面所通用的属性和方法抽离成一个BaseView&#xff0c;让其它页面继承该基础页面&#xff0c;同时将一些经常改变的属性和差异的属性写到配置文件里。例如树容器初始时是否展开、…

问题:下列哪些属于历史文化资源的特征( ). #学习方法#学习方法

问题&#xff1a;下列哪些属于历史文化资源的特征( ). A、稀缺性 B、脆弱性 C、可再生性 D、多样性 参考答案如图所示

C语言KR圣经笔记 7.1标准输入和输出 7.2格式化输出-printf

第七章 输入和输出 输入和输出功能并不是 C 语言本身的一部分&#xff0c;故到目前为止&#xff0c;本书都没有对其着重说明。然而&#xff0c;程序与其环境之间交互的方式&#xff0c;比书中之前所展示的更为复杂。本章我们会详描述标准库&#xff0c;即一系列为 C 程序提供输…

Stable Diffusion 模型下载:国风3 GuoFeng3

文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八案例九案例十推荐提示词下载地址模型介绍 欢迎使用GuoFeng3模型 - 这是一个中国华丽古风风格模型,也可以说是一个古风游戏角色模型,具有2.5D的质感。 条目内

数字人客服技术预研

技术洞察 引言 在当今数字化时代&#xff0c;不断进步和创新的人工智能&#xff08;AI&#xff09;技术已经渗透到各行各业中。随着AI技术、大模型技术逐步发展&#xff0c;使得数字人的广泛应用成为可能&#xff0c;本文将跟大家一起探讨AI数字人客服的概念、优势、应用场景…

【Java八股文面试系列】JVM-内存区域

目录 Java内存区域 运行时数据区域 线程独享区域 程序计数器 Java 虚拟机栈 StackFlowError&OOM 本地方法栈 线程共享区域 堆 GCR-分代回收算法 字符串常量池 方法区 运行时常量池 HotSpot 虚拟机对象探秘 对象的创建 对象的内存布局 句柄 Java内存区域 运…

使用WPS制作三线表

点击边框和底纹点击1、2、3、4并且应用于表格点击确定 再次选中表格点击右键表格属性选择边框和底纹 选中表格第一行右键点击表格属性选择边框和底纹 如果表格中存在虚线

左旋字符串的三种方法,并判断一个字符串是否为另外一个字符串旋转之后的字符串。(strcpy,strncat,strcmp,strstr函数的介绍)

一. 实现一个函数&#xff0c;可以左旋字符串中的k个字符。 例如&#xff1a; ABCD左旋一个字符得到BCDA ABCD左旋两个字符得到CDAB 通过分析&#xff0c;可以知道实际的旋转次数&#xff0c;其实是k%&#xff08;字符串长度&#xff09;。假设一个字…