当前位置: 首页 > news >正文

【动态规划】583. 两个字符串的删除操作、72. 编辑距离

提示:努力生活,开心、快乐的一天

文章目录

  • 583. 两个字符串的删除操作
    • 💡解题思路
    • 🤔遇到的问题
    • 💻代码实现
    • 🎯题目总结
  • 72. 编辑距离
    • 💡解题思路
    • 🤔遇到的问题
    • 💻代码实现
    • 🎯题目总结
  • 🎈今日心得


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

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

💡解题思路

  1. 本题和动态规划:115.不同的子序列 (opens new window)相比,其实就是两个字符串都可以删除了,情况虽说复杂一些,但整体思路是不变的
  2. 动规五部曲
  • 确定dp数组以及下标的含义:dp[i][j]:以i-1为结尾的字符串word1,和以j-1位结尾的字符串word2,想要达到相等,所需要删除元素的最少次数。
  • 确定递推公式:主要就是两大情况: 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});
    因为 dp[i][j - 1] + 1 = dp[i - 1][j - 1] + 2,所以递推公式可简化为:dp[i][j] = min(dp[i - 1][j] + 1, dp[i][j - 1] + 1);
    从字面上理解 就是 当 同时删word1[i - 1]和word2[j - 1],dp[i][j-1] 本来就不考虑 word2[j - 1]了,那么我在删 word1[i - 1],是不是就达到两个元素都删除的效果,即 dp[i][j-1] + 1。
  • dp数组如何初始化:dp[i][0]:word2为空字符串,以i-1为结尾的字符串word1要删除多少个元素,才能和word2相同呢,很明显dp[i][0] = i。
    同理dp[0][j] = j
  • 确定遍历顺序:从递推公式,可以看出,有三个方向可以推出dp[i][j]在这里插入图片描述
  • 举例推导dp数组:按照递推公式推导一下做推导,如果发现结果不对,就把dp数组打印出来在这里插入图片描述

🤔遇到的问题

  1. 注意初始化

💻代码实现

动态规划

var minDistance = function(word1, word2) {//dp[i][j],以i-1结尾的word1和以j-1结尾的word2,想要达到相等,所需要删除元素的最少次数。let w1 = word1.lengthlet w2 = word2.lengthlet dp = new Array(w1 + 1).fill(0).map(x => new Array(w2 + 1).fill(0))//初始化//word2为空字符串for (let i = 0; i <= w1; i++){dp[i][0] = i}//word1为空字符串for (let j = 0; j <= w2; j++){dp[0][j] = j}for (let i = 1; i <= w1; i++){for (let j = 1; j <= w2; j++){if (word1[i - 1] === word2[j - 1]) {dp[i][j] = dp[i-1][j-1]} else {dp[i][j] = Math.min(dp[i-1][j]+1,dp[i][j-1]+1)}}}console.log(dp)return dp[w1][w2]
};

🎯题目总结

特别注意的是:此题两个字符床都可以进行删除,所以在初始化和递推公式都会与之前有所不同,需要特别注意


72. 编辑距离

题目链接:72. 编辑距离

💡解题思路

  1. 动规五部曲
  • 确定dp数组以及下标的含义:dp[i][j] 表示以下标i-1为结尾的字符串word1,和以下标j-1为结尾的字符串word2,最近编辑距离为dp[i][j]
  • 确定递推公式:要考虑清楚编辑的几种操作,4种情况
if (word1[i - 1] == word2[j - 1])不操作
if (word1[i - 1] != word2[j - 1])增删换

情况1: if (word1[i - 1] == word2[j - 1]) 那么说明不用任何编辑,dp[i][j] 就应该是 dp[i - 1][j - 1],即dp[i][j] = dp[i - 1][j - 1];根据dp[i][j]的定义,word1[i - 1] 与 word2[j - 1]相等了,那么就不用编辑了,以下标i-2为结尾的字符串word1和以下标j-2为结尾的字符串word2的最近编辑距离dp[i - 1][j - 1]就是 dp[i][j]了。
情况2:if (word1[i - 1] != word2[j - 1])时
1、操作一:word1删除一个元素,那么就是以下标i - 2为结尾的word1 与 j-1为结尾的word2的最近编辑距离 再加上一个操作。即 dp[i][j] = dp[i - 1][j] + 1;
2、操作二:word2删除一个元素,那么就是以下标i - 1为结尾的word1 与 j-2为结尾的word2的最近编辑距离 再加上一个操作。即 dp[i][j] = dp[i][j - 1] + 1;
注意:word2添加一个元素,相当于word1删除一个元素,例如 word1 = “ad” ,word2 = “a”,word1删除元素’d’ 和 word2添加一个元素’d’,变成word1=“a”, word2=“ad”, 最终的操作数是一样!
3、操作三:替换元素,word1替换word1[i - 1],使其与word2[j - 1]相同,此时不用增删加元素。if (word1[i - 1] == word2[j - 1])的时候我们的操作 是 dp[i][j] = dp[i - 1][j - 1] 对吧。那么只需要一次替换的操作,就可以让 word1[i - 1] 和 word2[j - 1] 相同。所以 dp[i][j] = dp[i - 1][j - 1] + 1;
综上,当 if (word1[i - 1] != word2[j - 1]) 时取最小的,即:dp[i][j] = min({dp[i - 1][j - 1], dp[i - 1][j], dp[i][j - 1]}) + 1;

  • dp数组如何初始化:
    dp[i][0] :以下标i-1为结尾的字符串word1,和空字符串word2,最近编辑距离为dp[i][0]。那么dp[i][0]就应该是i,对word1里的元素全部做删除操作,即:dp[i][0] = i;同理dp[0][j] = j;
  • 确定遍历顺序:dp[i][j]是依赖左方,上方和左上方元素的在这里插入图片描述
  • 举例推导dp数组:按照递推公式推导一下做推导,如果发现结果不对,就把dp数组打印出来在这里插入图片描述

🤔遇到的问题

  1. word1[i - 1] != word2[j - 1时的三种情况的分析

💻代码实现

动态规划

var minDistance = function(word1, word2) {let w1 = word1.lengthlet w2 = word2.lengthlet dp = new Array(w1 + 1).fill(0).map(x => new Array(w2 + 1).fill(0))for (let i = 0; i <= w1; i++){dp[i][0] = i}for (let j = 0; j <= w2; j++){dp[0][j] = j}for (let i = 1; i <= w1; i++){for (let j = 1; j <= w2; j++){if (word1[i - 1] === word2[j - 1]) {dp[i][j] = dp[i-1][j-1]} else {dp[i][j] = Math.min(dp[i-1][j-1],dp[i-1][j],dp[i][j-1])+1}}}return dp[w1][w2]
};

🎯题目总结

if (word1[i - 1] != word2[j - 1])时
1、操作一:word1删除一个元素,那么就是以下标i - 2为结尾的word1 与 j-1为结尾的word2的最近编辑距离 再加上一个操作。即 dp[i][j] = dp[i - 1][j] + 1;
2、操作二:word2删除一个元素,那么就是以下标i - 1为结尾的word1 与 j-2为结尾的word2的最近编辑距离 再加上一个操作。即 dp[i][j] = dp[i][j - 1] + 1;
注意:word2添加一个元素,相当于word1删除一个元素,例如 word1 = “ad” ,word2 = “a”,word1删除元素’d’ 和 word2添加一个元素’d’,变成word1=“a”, word2=“ad”, 最终的操作数是一样!
3、操作三:替换元素,word1替换word1[i - 1],使其与word2[j - 1]相同,此时不用增删加元素。if (word1[i - 1] == word2[j - 1])的时候我们的操作 是 dp[i][j] = dp[i - 1][j - 1] 对吧。那么只需要一次替换的操作,就可以让 word1[i - 1] 和 word2[j - 1] 相同。所以 dp[i][j] = dp[i - 1][j - 1] + 1;
综上,当 if (word1[i - 1] != word2[j - 1]) 时取最小的,即:dp[i][j] = min({dp[i - 1][j - 1], dp[i - 1][j], dp[i][j - 1]}) + 1;

🎈今日心得

编辑距离的题目终于达到了最难的一题,确实难

相关文章:

【动态规划】583. 两个字符串的删除操作、72. 编辑距离

提示&#xff1a;努力生活&#xff0c;开心、快乐的一天 文章目录 583. 两个字符串的删除操作&#x1f4a1;解题思路&#x1f914;遇到的问题&#x1f4bb;代码实现&#x1f3af;题目总结 72. 编辑距离&#x1f4a1;解题思路&#x1f914;遇到的问题&#x1f4bb;代码实现&…...

Gradient conjugate priors and multi-layer neural networks

动机 先验参数 m , α , β , v m,\alpha,\beta,v m,α,β,v和随机变量 τ \tau τ KL散度的形式是&#xff1a; Dynamics of m , α , β , v m,\alpha,\beta,v m,α,β,v Dynamics of m , β , v m,\beta,v m,β,v for a fixed α \alpha α 绿色轨迹连接初始点和目标点…...

DistributedDataParallel数据不均衡

背景 在使用 DistributedDataParallel 进行数据并行训练时&#xff0c;每次反向传播都需要执行 all_reduce 操作以同步各个进程的梯度。all_reduce 需要进程组中的所有进程参与&#xff0c;如果某一个进程没有执行 all_reduce&#xff08;一个进程的输入较其他进程少&#xff…...

Cloud Studio连接MySQL,Access denied for一系列问题

官方文档有写如何安装Mysql $ apt update $ apt install mysql-server mysql-client -y$ service mysql start mysql -uroot -p123456进入MySQL命令行 问题出在连接数据库这一步&#xff0c;命令行能进去&#xff0c;但是数据库插件和代码都连不上 Access denied for 大概率…...

经典题型---旋转数组

经典题型—旋转数组 文章目录 经典题型---旋转数组一、题目二、代码实现 一、题目 给定一个整数数组 nums&#xff0c;将数组中的元素向右轮转 k 个位置&#xff0c;其中 k 是非负数。 示例 1: 输入: nums [1,2,3,4,5,6,7], k 3 输出: [5,6,7,1,2,3,4] 解释: 向右轮转 1 步…...

vue如何实现登录数据的持久化

Vue.js是一款流行的JavaScript框架&#xff0c;它可以帮助开发者构建高效且易于维护的单页面应用程序。在Vue.js中&#xff0c;实现登录数据的持久化是一个重要的任务&#xff0c;因为它可以帮助用户保持登录状态并避免频繁的登录操作。在本文中&#xff0c;我们将讨论Vue.js如…...

【Unity3D编辑器开发】Unity3D中实现Transform组件拓展,快速复制、粘贴、复原【非常实用】

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址我的个人博客 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 在开发中&#xff0c;常常会遇到频繁复制粘贴物体的坐标、旋转…...

求解仿射变换矩阵

仿射变换是图形学中经常用到的方法&#xff0c;通常但是仿射变换的系数是未知的&#xff0c;需要找到变换前后的三对对应点进行求解。 from affine import Affine import numpy as np参考文献 矩阵最小二乘法求解仿射变换矩阵 def solve_affine(init_points, goal_points) -&…...

【每日一题】—— 最大素因子

&#x1f30f;博客主页&#xff1a;PH_modest的博客主页 &#x1f6a9;当前专栏&#xff1a;每日一题 &#x1f48c;其他专栏&#xff1a; &#x1f534; 每日反刍 &#x1f7e1; C跬步积累 &#x1f7e2; C语言跬步积累 &#x1f308;座右铭&#xff1a;广积粮&#xff0c;缓称…...

【JavaEE】JUC 常见的类 -- 多线程篇(8)

JUC 常见的类 1. Callable 接口2. ReentrantLock3. 原子类4. 线程池5. 信号量 Semaphore6. CountDownLatch 1. Callable 接口 Callable Interface 也是一种创建线程的方式 Runnable 能表示一个任务 (run方法) – 返回 voidCallable 也能表示一个任务(call方法) 返回一个具体的…...

java项目运行时信息获取

大体思路如下&#xff0c;想要获取启动时处理器数量、jvm 相关信息&#xff0c;操作系统信息、运行机器信息 运行机器信息 import org.slf4j.Logger; import org.slf4j.LoggerFactory;import java.lang.invoke.MethodHandles;/*** 机器工具类*/ public abstract class ServerU…...

【LeetCode】71. 简化路径

1 问题 给你一个字符串 path &#xff0c;表示指向某一文件或目录的 Unix 风格 绝对路径 &#xff08;以 / 开头&#xff09;&#xff0c;请你将其转化为更加简洁的规范路径。 在 Unix 风格的文件系统中&#xff0c;一个点&#xff08;.&#xff09;表示当前目录本身&#xf…...

操作系统【OS】进程的控制【进程的创建、终止、阻塞、唤醒】

定义和过程 对应事件 创建 允许一个进程创建另一个进程允许子进程继承父进程所拥有的资源创建进程的过程如下&#xff1a; 申请一个空白的 PCB&#xff0c;并向 PCB 中填写一些控制和管理进程的信息&#xff0c;比如进程的唯一标识等&#xff1b;为该进程分配运行时所必需的…...

写一个简单的解释器(2) 构建标记流

确定标记类型 分为几个大类&#xff1a; 用户符号&#xff08;类型/标识符/数字/字符串…)关键字 (流程控制和定义符)括号 &#xff08;这里暂时认为 [] 属于括号&#xff09;分号 上述四类标记基本囊括了 vc \texttt{vc} vc 中的所有最小单元的类型&#xff0c;但是因为构…...

Leetcode1833. 雪糕的最大数量

Every day a Leetcode 题目来源&#xff1a;1833. 雪糕的最大数量 解法1&#xff1a;贪心 排序 本题唯一的难点在于计数排序。 计数排序详解&#xff1a;C算法之计数排序 为了尽可能多的买到雪糕&#xff0c;我们选择从价格低的雪糕开始买&#xff0c;统计能够买到的雪糕…...

idea 里 没有svn选项的处理办法

总结一下没有svn选项的几种情况&#xff1a; 情况1&#xff1a;IntelliJ IDEA打开带SVN信息的项目不显示SVN信息&#xff0c;项目右键SVN以及图标还有Changes都不显示解决方法 在VCS菜单中有个开关&#xff0c;叫Enabled Version Control Integration&#xff0c;在打开的窗口…...

基于SpringBoot的招生管理系统

基于SpringBoot的招生管理系统的设计与实现~ 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBootMyBatisVue工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 主页 登录界面 管理员界面 用户界面 摘要 基于SpringBoot的招生管理系统是一款现…...

01、MySQL-------性能优化

目录 一、影响性能的相关因素存储过程&#xff1a; 二、sql优化1>、Mysql系统架构2>、引擎区别&#xff1a; 3>、索引1、什么是索引&#xff1f;联合主键索引理解&#xff1a;索引长度理解&#xff1a;什么是慢查询&#xff1f; 1&#xff09;、索引理解2&#xff09;…...

Flutter - APP跳转高德、百度、腾讯、谷歌地图

demo 地址: https://github.com/iotjin/jh_flutter_demo 代码不定时更新&#xff0c;请前往github查看最新代码 这里介绍的是不需要自己开发地图&#xff0c;直接通过给定的经纬度&#xff0c;跳转到三方地图APP调用导航的方式 一种是写的工具类&#xff0c;一种是通过调用三方…...

Flyway Desktop updated

Flyway Desktop updated 为比较工件序列化和反序列化添加了额外的调试日志记录。 Flyway Desktop现在将记住以前用于创建项目和匹配克隆的位置。 新的脱机许可工作流现在已在Microsoft Windows上启用。 现在&#xff0c;在配置目标数据库列表时&#xff0c;环境ID是可见的。 现…...

TranslucentTB启动失败?3步解决Microsoft.UI.Xaml.2.8缺失问题终极指南

TranslucentTB启动失败&#xff1f;3步解决Microsoft.UI.Xaml.2.8缺失问题终极指南 【免费下载链接】TranslucentTB A lightweight utility that makes the Windows taskbar translucent/transparent. 项目地址: https://gitcode.com/gh_mirrors/tr/TranslucentTB Trans…...

xilinx vivado cameralink图像接收与发送代码,最大支持并行速度100MH...

xilinx vivado cameralink图像接收与发送代码&#xff0c;最大支持并行速度100MHz&#xff0c;优于编解码接口芯片。 不利用解码与编码芯片&#xff0c;直接在FPGA内部进行接收解码和发送。1. 系统架构总览 1.1 设计背景与目标 本代码实现了一个完整的Camera Link接口解决方案…...

QuickLook OfficeViewer-Native:基于原生Office组件的文档预览解决方案

QuickLook OfficeViewer-Native&#xff1a;基于原生Office组件的文档预览解决方案 【免费下载链接】QuickLook.Plugin.OfficeViewer-Native View Word, Excel, and PowerPoint files with MS Office and WPS Office components. 项目地址: https://gitcode.com/gh_mirrors/q…...

DownKyi终极指南:5步掌握B站8K超高清视频下载的完整方法

DownKyi终极指南&#xff1a;5步掌握B站8K超高清视频下载的完整方法 【免费下载链接】downkyi 哔哩下载姬downkyi&#xff0c;哔哩哔哩网站视频下载工具&#xff0c;支持批量下载&#xff0c;支持8K、HDR、杜比视界&#xff0c;提供工具箱&#xff08;音视频提取、去水印等&…...

从连续到数字:深入解读Matlab离散化函数c2d的6种方法及其在滤波器与控制器设计中的选用

从连续到数字&#xff1a;Matlab离散化函数c2d的6种方法及其在工程实践中的精准选用 在数字信号处理和控制系统的设计中&#xff0c;连续时间系统的离散化是一个无法绕开的关键环节。就像摄影师需要将现实世界的连续光影转化为数码相机中的像素一样&#xff0c;工程师也需要将连…...

别再只盯着RTP了!用Wireshark抓包实战,5分钟看懂RTCP的SR和RR报告到底在说啥

别再只盯着RTP了&#xff01;用Wireshark抓包实战&#xff0c;5分钟看懂RTCP的SR和RR报告到底在说啥 当你在调试视频会议卡顿或直播延迟问题时&#xff0c;是否曾盯着Wireshark里密密麻麻的RTP包感到无从下手&#xff1f;其实&#xff0c;解决问题的关键往往藏在那些被忽略的RT…...

Vue3-Marquee 架构深度解析:零依赖跑马灯组件的设计哲学与实践

Vue3-Marquee 架构深度解析&#xff1a;零依赖跑马灯组件的设计哲学与实践 【免费下载链接】vue3-marquee A simple marquee component with ZERO dependencies for Vue 3. 项目地址: https://gitcode.com/gh_mirrors/vu/vue3-marquee 在 Vue 3 生态系统中&#xff0c;动…...

别再手动调图了!用MATLAB代码批量美化论文折线图(附完整参数设置清单)

MATLAB科研绘图革命&#xff1a;代码化美学设计全指南 科研图表是论文的"门面"&#xff0c;但多数研究者仍被困在重复的手动调整中。我曾耗时两周调整30组实验数据的图表格式&#xff0c;直到发现代码化美学的力量——现在只需5分钟就能完成过去两天的工作量。 1. 为…...

避坑指南:ESP32 MicroPython读写SD卡,为什么你的代码总报错?

ESP32 MicroPython SD卡读写避坑实战&#xff1a;从报错到稳定运行的深度解析 当你在ESP32上尝试用MicroPython操作SD卡时&#xff0c;是否遇到过这些令人抓狂的场景&#xff1f;明明按照教程连接了硬件&#xff0c;代码却抛出OSError: no SD card&#xff1b;或者文件系统挂载…...

手把手教你排查ROS Noetic下的TF_REPEATED_DATA警告:从roswtf工具到源码定位

深度解析ROS Noetic中TF_REPEATED_DATA警告的排查与修复 当你在Ubuntu 20.04上运行ROS Noetic时&#xff0c;突然发现终端不断刷出"TF_REPEATED_DATA ignoring data with redundant timestamp for frame"的警告信息&#xff0c;同时Rviz中的机器人模型出现异常抖动—…...