代码随想录算法训练营Day56|583. 两个字符串的删除操作、72. 编辑距离

本文主要是介绍代码随想录算法训练营Day56|583. 两个字符串的删除操作、72. 编辑距离,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

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

前言

思路

算法实现 

法二

72. 编辑距离

前言

思路

算法实现 

总结


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

题目链接

文章链接

前言

        本题与上一题不同的子序列相比,变化就是两个字符串都可以进行删除操作了。

思路

         利用动规五部曲进行分析:

1.确定dp数组及其下标的含义:

        dp[i][j]:以i-1为结尾的字符串word1,和以j-1位结尾的字符串word2,想要达到相等,所需要删除元素的最少次数。

2.确定递推公式:

        递推公式的推导与前几题大致类似,都有分两种情况进行讨论:

  • 当word1[i - 1] 与 word2[j - 1]相同的时候;
  • 当word1[i - 1] 与 word2[j - 1]不相同的时候

        对于word1[i - 1] 与 word2[j - 1]相同时,dp[i][j] = dp[i - 1][j - 1];

        当word1[i - 1] 与 word2[j - 1]不相同的时候,有三种情况:

        情况一:删word1[i - 1],最少操作次数为dp[i - 1][j] + 1,

        情况二:删word2[j - 1],最少操作次数为dp[i][j - 1] + 1,

        情况三,同时删word1[i - 1]和word2[j - 1], 操作的最少次数为dp[i - 1][j - 1] + 2;

        最终结果是取三种情况的最小值,dp[i][j] = min({dp[i - 1][j - 1] + 2, dp[i - 1][j] + 1, dp[i][j - 1] + 1});

3.初始化dp数组:

        从递推公式中,可以看出来,dp[i][0] 和 dp[0][j]是一定要初始化的。

        当word2为空字符串时,word1字符串的长度为i,因此要删i次才能与空字符串word2相等,所以dp[i][0]的初值为i,同理dp[0][j]的初值为j;

4.确定遍历顺序:

        从递推公式 dp[i][j] = min(dp[i - 1][j - 1] + 2, min(dp[i - 1][j], dp[i][j - 1]) + 1); 和dp[i][j] = dp[i - 1][j - 1]可以看出dp[i][j]都是根据左上方、正上方、正左方推出来的。

        所以遍历的时候一定是从上到下,从左到右,这样保证dp[i][j]可以根据之前计算出来的数值进行计算。

5.打印dp数组:

        以word1:"sea",word2:"eat"为例,推导dp数组状态图如下:

算法实现 

class Solution {
public:int minDistance(string word1, string word2) {vector<vector<int>> dp(word1.size() + 1, vector<int> (word2.size() + 1, 0));for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;for (int j = 1; j <= word2.size(); j++) dp[0][j] = j;for (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] + 1, min(dp[i][j - 1] + 1, dp[i - 1][j - 1] + 2));}}}return dp[word1.size()][word2.size()];}
};

法二

        利用求最长公共子序列的思想,两个字符串要删除的部分就是最长公共子序列之外的部分。

class Solution {
public:int minDistance(string word1, string word2) {vector<vector<int>> dp(word1.size() + 1, vector<int> (word2.size() + 1, 0));for (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] + 1;else dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);}}return word1.size() + word2.size() - dp[word1.size()][word2.size()] * 2;}
};

72. 编辑距离

题目链接

文章链接

前言

         前几题都是为了本题做铺垫,有了前面几题的学习接触本题就不会觉得非常困难,主要难点还是在于递推公式的确定,尤其是当两个字符串比较的位置字符不相等时递推公式的确定。

思路

         还是利用动规五部曲进行分析:

1.确定dp数组及其下标的含义:

        dp[i][j]:以下标i-1为结尾的字符串word1,和以下标j-1为结尾的字符串word2,最近编辑距离为dp[i][j]。

2.确定递推公式:

        依然是分两种大情况进行讨论:

  • 当word1[i - 1] 与 word2[j - 1]相同;
  • 当word1[i - 1] 与 word2[j - 1]不相同;

        当word1[i - 1] 与 word2[j - 1]相同时,不需要进行额外的操作(编辑距离),和word1以i - 2为结尾,word2以就j - 2为结尾要操作的次数一样,即dp[i][j] = dp[i - 1][j - 1];

        而当word1[i - 1] 与 word2[j - 1]不相同时,要分别考虑删、增、换三种不同的情况;

        增删元素其实本质上是一样的,在word1中增加元素和在word2中删除元素起到的效果相同,此时dp[i][j] = dp[i - 1][j] + 1(删word1中的元素),或者dp[i][j] = dp[i][j - 1] + 1(删除word2中的元素);

        替换元素时,替换word[i - 1]元素使其与word2[j - 1]相等(也可以倒过来),此时dp[i][j] = dp[i - 1][j - 1] + 1;

3.dp数组初始化

        与上题一样dp[i][0] = i,dp[0][j] = j,只需要删除完所有字符就能与另一个空字符串相等;

4.确定遍历顺序:

        从递推公式可以看出,dp[i][j]是依赖左方,上方和左上方元素的,如图:

        

5.打印dp数组:

        以示例1为例,输入:word1 = "horse", word2 = "ros"为例,dp矩阵状态图如下:

         

算法实现 

class Solution {
public:int minDistance(string word1, string word2) {vector<vector<int>> dp(word1.size() + 1, vector<int> (word2.size() + 1, 0));for (int i = 1; i <= word1.size(); i++) dp[i][0] = i;for (int j = 1; j <= word2.size(); j++) dp[0][j] = j;for (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] + 1, min(dp[i][j - 1] + 1, dp[i - 1][j - 1] + 1));}}return dp[word1.size()][word2.size()];}
};

总结

        今天的两道题是前面几道题的深化,循序渐进。

这篇关于代码随想录算法训练营Day56|583. 两个字符串的删除操作、72. 编辑距离的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

Python中OpenCV与Matplotlib的图像操作入门指南

《Python中OpenCV与Matplotlib的图像操作入门指南》:本文主要介绍Python中OpenCV与Matplotlib的图像操作指南,本文通过实例代码给大家介绍的非常详细,对大家的学... 目录一、环境准备二、图像的基本操作1. 图像读取、显示与保存 使用OpenCV操作2. 像素级操作3.

IIS 7.0 及更高版本中的 FTP 状态代码

《IIS7.0及更高版本中的FTP状态代码》本文介绍IIS7.0中的FTP状态代码,方便大家在使用iis中发现ftp的问题... 简介尝试使用 FTP 访问运行 Internet Information Services (IIS) 7.0 或更高版本的服务器上的内容时,IIS 将返回指示响应状态的数字代

MySQL 添加索引5种方式示例详解(实用sql代码)

《MySQL添加索引5种方式示例详解(实用sql代码)》在MySQL数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中,下面给大家分享MySQL添加索引5种方式示例详解(实用sql代码),... 在mysql数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中。索引可以在创建表时定义,也可

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Python实现一键PDF转Word(附完整代码及详细步骤)

《Python实现一键PDF转Word(附完整代码及详细步骤)》pdf2docx是一个基于Python的第三方库,专门用于将PDF文件转换为可编辑的Word文档,下面我们就来看看如何通过pdf2doc... 目录引言:为什么需要PDF转Word一、pdf2docx介绍1. pdf2docx 是什么2. by

Spring Security介绍及配置实现代码

《SpringSecurity介绍及配置实现代码》SpringSecurity是一个功能强大的Java安全框架,它提供了全面的安全认证(Authentication)和授权(Authorizatio... 目录简介Spring Security配置配置实现代码简介Spring Security是一个功能强

python操作redis基础

《python操作redis基础》Redis(RemoteDictionaryServer)是一个开源的、基于内存的键值对(Key-Value)存储系统,它通常用作数据库、缓存和消息代理,这篇文章... 目录1. Redis 简介2. 前提条件3. 安装 python Redis 客户端库4. 连接到 Re

通过cmd获取网卡速率的代码

《通过cmd获取网卡速率的代码》今天从群里看到通过bat获取网卡速率两段代码,感觉还不错,学习bat的朋友可以参考一下... 1、本机有线网卡支持的最高速度:%v%@echo off & setlocal enabledelayedexpansionecho 代码开始echo 65001编码获取: >

Java集成Onlyoffice的示例代码及场景分析

《Java集成Onlyoffice的示例代码及场景分析》:本文主要介绍Java集成Onlyoffice的示例代码及场景分析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 需求场景:实现文档的在线编辑,团队协作总结:两个接口 + 前端页面 + 配置项接口1:一个接口,将o

Java Stream.reduce()方法操作实际案例讲解

《JavaStream.reduce()方法操作实际案例讲解》reduce是JavaStreamAPI中的一个核心操作,用于将流中的元素组合起来产生单个结果,:本文主要介绍JavaStream.... 目录一、reduce的基本概念1. 什么是reduce操作2. reduce方法的三种形式二、reduce