算法日记day 44(动归之编辑距离|回文字串|最长回文子序列)

news2024/12/26 20:47:24

一、编辑距离

题目:

给你两个单词 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')

思路:

dp数组的含义是以i-1为结尾的字符串word1和以j-1长度为结尾的字符串word2所需要的步骤为dp[i][j],有四种情况,如果本身两字符串所对应的元素相同,则不需要做任何操作,因此

                                                         dp[i][j] = dp[i-1][j-1]

如果需要对word1进行删除操作,相当于忽略掉word1中的第i-1位元素,这样的

                                                        dp[i][j] = dp[i-1][j] + 1

如果需要对word1进行添加操作,相当于对word2进行一次删除操作,例如word1="ab"  word2="a",可以删除word1中的b,或者添加word2中一位b,因此

                                                        dp[i][j] = dp[i][j-1] + 1

如果进行修改的操作,相当于字符串中其前i-2位和j-2位元素均相同,仅需改变其中i-1(j-1)位元素,因此

                                                         dp[i][j] = dp[i-1][j-1] + 1

对于需要操作的元素,应取三者中次数最少的值作为最终编辑距离

初始化操作,对于dp[i][0],相当于长度为 i 的字符串word1转换为长度为0的字符串word2,仅需删除操作,次数为字符串word1的长度,因此

                                                            dp[i][0] = i

同理:                                                 dp[0][j] = j 

代码:

 

public int minDistance(String word1, String word2) {
    // 初始化 dp 数组,dp[i][j] 表示将 word1 的前 i 个字符转换为 word2 的前 j 个字符所需的最小操作数
    int[][] dp = new int[word1.length() + 1][word2.length() + 1];
    
    // 设置 dp 数组的第一列,dp[i][0] 表示将 word1 的前 i 个字符转换为空字符串的操作数
    for (int i = 1; i <= word1.length(); i++)
        dp[i][0] = i;
    
    // 设置 dp 数组的第一行,dp[0][j] 表示将空字符串转换为 word2 的前 j 个字符的操作数
    for (int j = 1; j < word2.length() + 1; j++)
        dp[0][j] = j;
    
    // 填充 dp 数组
    for (int i = 1; i < word1.length() + 1; i++) {
        for (int j = 1; j <= word2.length(); j++) {
            // 如果当前字符相同,不需要增加额外的操作,继承前一个状态的值
            if (word1.charAt(i - 1) == word2.charAt(j - 1)) {
                dp[i][j] = dp[i - 1][j - 1];
            } else {
                // 如果字符不同,考虑三种操作:
                // 1. 替换字符:dp[i - 1][j - 1] + 1
                // 2. 删除字符:dp[i - 1][j] + 1
                // 3. 插入字符:dp[i][j - 1] + 1
                // 选择三者中的最小值作为当前状态的值
                dp[i][j] = Math.min(dp[i - 1][j - 1], Math.min(dp[i - 1][j], dp[i][j - 1])) + 1;
            }
        }
    }
    
    // 返回将 word1 转换为 word2 所需的最小操作数
    return dp[word1.length()][word2.length()];
}
  1. 初始化 dp 数组

    • dp[i][j] 表示将 word1 的前 i 个字符转换为 word2 的前 j 个字符的最小编辑距离。
  2. 第一列和第一行的初始化

    • dp[i][0] 表示将 word1 的前 i 个字符转换为空字符串的操作数,即删除操作的数量。
    • dp[0][j] 表示将空字符串转换为 word2 的前 j 个字符的操作数,即插入操作的数量。
  3. 填充 dp 数组

    • 遍历所有可能的子问题,并根据当前字符是否相同来决定最小的操作数。
    • 如果字符相同,继承前一个状态的值;如果字符不同,则选择替换、删除或插入操作中最小的代价。
  4. 返回结果

    • 最终的编辑距离是将 word1 的所有字符转换为 word2 的所有字符所需的最小操作数。

二、回文字串 

题目:

给你一个字符串 s ,请你统计并返回这个字符串中 回文子串 的数目。

回文字符串 是正着读和倒过来读一样的字符串。

子字符串 是字符串中的由连续字符组成的一个序列。

示例 1:

输入:s = "abc"
输出:3
解释:三个回文子串: "a", "b", "c"

示例 2:

输入:s = "aaa"
输出:6
解释:6个回文子串: "a", "a", "a", "aa", "aa", "aaa"

思路:

首先定义一个boolean类型的dp数组,本题中dp数组的含义是在区间[i,j]的字符串中是回文字串的dp[i][j]为true

判断

由于回文串的特殊性,如果 i 与 j 的值相同,当 i 与 j 的距离满足 j - i <= 2时,都符合回文串的条件因此定义一个res收集结果,同时dp[i][j]为true,如果此时距离大于2,则需要比较dp[i+1][j-1]的值是否为true,如果是,则说明dp[i][j]是回文串,如果 i 与 j 的值不相同,则表示不是回文串

代码:

public int countSubstrings(String s) {
    // 将字符串转换为字符数组,方便使用索引访问字符
    char[] chars = s.toCharArray();
    int len = chars.length; // 获取字符数组的长度
    // 创建一个二维布尔数组 dp,dp[i][j] 表示子串 chars[i..j] 是否为回文
    boolean[][] dp = new boolean[len][len];
    int result = 0; // 结果变量,用于记录回文子串的总数量

    // 从字符串的末尾开始,逐步向前遍历所有可能的起始位置
    for (int i = len - 1; i >= 0; i--) {
        // 对于每个起始位置 i,遍历可能的结束位置 j
        for (int j = i; j < len; j++) {
            // 检查当前子串 chars[i..j] 的首尾字符是否相同
            if (chars[i] == chars[j]) {
                // 当子串长度为 1 或 2 时,直接标记为回文子串
                // 1. 长度为 1 的子串(例如 "a")
                // 2. 长度为 2 的子串(例如 "aa")
                if (j - i <= 2) { // 子串长度小于或等于 2
                    result++; // 增加回文子串计数
                    dp[i][j] = true; // 标记 dp[i][j] 为回文
                } 
                // 当子串长度大于 2 时,检查内部子串是否为回文
                // 3. 如果子串 chars[i+1..j-1] 为回文,则 chars[i..j] 也是回文
                else if (dp[i + 1][j - 1]) { 
                    result++; // 增加回文子串计数
                    dp[i][j] = true; // 标记 dp[i][j] 为回文
                }
            }
        }
    }

    return result; // 返回总的回文子串数量
}
  • len 是字符串的长度。
  • dp 是一个二维布尔数组,dp[i][j] 用来记录子串 chars[i..j] 是否为回文。
  • result 用来累计回文子串的总数。
  • 外层循环从字符串的末尾向前遍历起始位置 i
  • 内层循环从起始位置 i 遍历到字符串的末尾,考虑所有可能的结束位置 j
  • 首先,检查 chars[i] 是否等于 chars[j]。如果不相等,chars[i..j] 不能是回文子串。
  • 对于长度为 1 或 2 的子串(即 j - i <= 2),直接判断为回文。
  • 对于长度大于 2 的子串,需要检查内部子串 chars[i+1..j-1] 是否是回文。如果是,则 chars[i..j] 也是回文。
  • 返回计算得到的回文子串总数。

三、最长回文子序列 

题目:

给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。

子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。

示例 1:

输入:s = "bbbab"
输出:4
解释:一个可能的最长回文子序列为 "bbbb" 。

示例 2:

输入:s = "cbbd"
输出:2
解释:一个可能的最长回文子序列为 "bb" 。

思路:

首先dp数组的含义是在区间为[i,j]的字符串中最长回文子序列的长度为dp[i][j]

对于 i 与 j 相同的情况,若区间内是回文序列,则相对应的最长长度加2

                                                     dp[i][j] = dp[i+1][j-1] +2

如果 i 与 j 不相同,则取dp[i][j-1] 与dp[i-1][j] 中的最大值,即为

                                   dp[i][j] = max(dp[i][j],max(dp[i-1][j],dp[i][j-1])

又dp[i][j] 依赖于 dp[i + 1][j - 1] ,dp[i + 1][j] 和 dp[i][j - 1]

遍历顺序应为 i 从下往上,j 从左往右

 

代码:

public int longestPalindromeSubseq(String s) {
    int len = s.length(); // 获取字符串的长度
    // 创建一个二维数组 dp,其中 dp[i][j] 表示子串 s[i..j] 的最长回文子序列的长度
    int[][] dp = new int[len + 1][len + 1];

    // 初始化每个字符的最长回文子序列的长度为 1,因为任何单个字符都是回文的
    for (int i = 0; i < len; i++) {
        dp[i][i] = 1;
    }

    // 从字符串的倒数第二个字符开始,逐步向前遍历所有可能的起始位置
    for (int i = len - 1; i >= 0; i--) {
        // 对于每个起始位置 i,遍历可能的结束位置 j
        for (int j = i + 1; j < len; j++) {
            // 如果子串 s[i..j] 的首尾字符相同
            if (s.charAt(i) == s.charAt(j)) {
                // 则子串 s[i..j] 的最长回文子序列长度为子串 s[i+1..j-1] 的长度 + 2
                dp[i][j] = dp[i + 1][j - 1] + 2;
            } else {
                // 如果首尾字符不同,则子串 s[i..j] 的最长回文子序列长度是
                // 子串 s[i+1..j] 和 s[i..j-1] 的最长回文子序列长度中的较大值
                dp[i][j] = Math.max(dp[i + 1][j], dp[i][j - 1]);
            }
        }
    }

    // 返回整个字符串 s 的最长回文子序列的长度
    return dp[0][len - 1];
}
  • 外层循环从字符串的倒数第二个字符向前遍历所有起始位置 i
  • 内层循环从起始位置 i 向后遍历可能的结束位置 j
  • 如果 s[i] 和 s[j] 相同,则子串 s[i..j] 的最长回文子序列长度等于 s[i+1..j-1] 的最长回文子序列长度加 2(包括 s[i] 和 s[j])。
  • 如果 s[i] 和 s[j] 不同,则子串 s[i..j] 的最长回文子序列长度是 s[i+1..j] 和 s[i..j-1] 中较大的值。

 

今天的学习就到这里 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

WEB渗透免杀篇-cshot远程shellcode

往期文章 WEB渗透免杀篇-免杀工具全集-CSDN博客 WEB渗透免杀篇-加载器免杀-CSDN博客 WEB渗透免杀篇-分块免杀-CSDN博客 WEB渗透免杀篇-Powershell免杀-CSDN博客 WEB渗透免杀篇-Python源码免杀-CSDN博客 WEB渗透免杀篇-C#源码免杀-CSDN博客 WEB渗透免杀篇-MSFshellcode免杀…

基于.net技术的物业管理系统需求分析与设计

系统需求分析 2.1 整体需求概述 根据某XXXXXXXX管理公司实际业务调研分析&#xff0c;可将其系统需求划分为7个部分&#xff1a;基础信息维护、网上报修、权限管理、动力消耗、物料管理、收费管理、报表分析。 2.1.1 基础信息维护 基础信息维护包括对以下业务基础数据的采集…

linux驱动——设备树

1&#xff1a;初识设备树 1.1 什么是设备树&#xff0c;设备树的意义 设备树&#xff08;Device Tree&#xff09;是 Linux 内核中用于描述硬件设备的一种数据结构。它为操作系统提供了一种抽象的方法&#xff0c;使其能够识别和配置硬件设备&#xff0c;而无需将硬件细节硬编…

QT-五子棋游戏

QT-五子棋游戏 一、演示效果二、核心代码三、下载链接 一、演示效果 二、核心代码 #include "GameModel.h" #include <time.h> #include <stdlib.h>GameModel::GameModel(){}void GameModel::startGame(GameType type){gameType type;//初始化棋盤game…

【备忘录模式】设计模式系列:掌握状态回溯的艺术(设计详解)

文章目录 备忘录设计模式详解引言1. 设计模式概述2. 备忘录模式的基本概念2.1 备忘录模式的定义2.2 备忘录模式的关键角色 3. 备忘录模式的实现原理3.1 备忘录模式的工作流程3.2 模式的优缺点分析3.3 与其他模式的对比 4. 实际案例分析4.1 游戏状态保存与恢复4.2 文档编辑器撤销…

19529 照明灯安装

### 详细分析 这个问题可以通过二分查找和贪心算法来解决。我们需要找到一个最大值&#xff0c;使得在这个最大值下&#xff0c;能够在给定的坐标上安装 k 个照明灯&#xff0c;并且相邻的照明灯之间的距离至少为这个最大值。 ### 思路 1. **排序**&#xff1a;首先对给定的…

S3C2440中断处理

一、中断处理机制概述 中断是CPU在执行程序过程中&#xff0c;遇到急需处理的事件时&#xff0c;暂时停止当前程序的执行&#xff0c;转而执行处理该事件的中断服务程序&#xff0c;并在处理完毕后返回原程序继续执行的过程。S3C2440提供了丰富的中断源&#xff0c;包括内部中…

微信小程序:开发工具修改js编译后还是旧的js逻辑

1、清理所有缓存&#xff0c;重新导入项目 2、语法存在问题无法编译,导致内存堆积&#xff0c;无法自动编译 3、npm 存在问题&#xff0c;可以重新构建 4、有时候编译器也没报错都是一切正常&#xff0c;但是编译后依然不是最新。这个时候需要考虑下电脑是否存在问题&#xff0…

使用gitee存储项目

gitee地址&#xff1a;Gitee - 基于 Git 的代码托管和研发协作平台 创建gitee远程仓库 将远程仓库内容拉取到本地仓库 复制下面这个地址 通过小乌龟便捷推送拉取代码&#xff1a;https://blog.csdn.net/m0_65520060/article/details/140091437

Ubuntu | 解决 VMware 中 Ubuntu 虚拟机磁盘空间不足的问题

目录 一、存在的问题二、解决的步骤第一步&#xff1a;扩展磁盘空间第二步&#xff1a;查看磁盘空间使用情况第三步&#xff1a;安装分区工具第四步&#xff1a;启动分区工具第五步&#xff1a;修改挂载文件夹的读写权限第六步&#xff1a;扩展文件系统大小第七步&#xff1a;验…

Prometheus2:被监控机器安装node_exporter与配置

1. 下载node_exporter [rootlocalhost ~]# wget https://github.com/prometheus/node_exporter/releases/download/v1.8.2/node_exporter-1.8.2.linux-amd64.tar.gz 2. 解压缩 [rootlocalhost ~]# tar -zxvf node_exporter-1.8.2.linux-amd64.tar.gz 3. 复制到/usrl/local路…

sed命令用法与案例

在Linux操作系统中&#xff0c;sed&#xff08;stream editor&#xff09;是一种功能强大的文本处理工具&#xff0c;用于执行文本的查找、替换、删除、新增等操作。sed命令以其简洁的语法和高效的执行速度&#xff0c;在自动化脚本和文本处理中扮演着重要角色。本文将探讨sed命…

探索串行通信的奥秘:Python中的pyserial库

文章目录 探索串行通信的奥秘&#xff1a;Python中的pyserial库背景&#xff1a;为何选择pyserial&#xff1f;pyserial是什么&#xff1f;如何安装pyserial&#xff1f;pyserial的五个简单函数场景应用&#xff1a;pyserial在实际中的使用常见bug及解决方案总结 探索串行通信的…

HR招聘,如何解决招聘需求不明确的问题

在HR招聘过程中&#xff0c;遇到招聘需求不明确的问题时&#xff0c;可以通过一系列措施来明确需求&#xff0c;提高招聘效率和质量。同时&#xff0c;在线人才测评、职业性格测试、认知能力测试和心理健康测试等工具也可以作为辅助手段&#xff0c;帮助HR更准确地评估候选人。…

【大模型从入门到精通33】开源库框架LangChain RAG 系统中的问答技术3

这里写目录标题 理论问答过程的三个主要阶段传递文档片段至 LM 上下文窗口的局限性及策略向量数据库的重要性RetrievalQA 链的作用MapReduce 与 Refine 的区别分布式系统中的实际考量实验的重要性RetrievalQA 链的主要限制对话记忆的重要性 实践初始化向量数据库设置 Retrieval…

GD32双路CAN踩坑记录

GD32双路CAN踩坑记录 目录 GD32双路CAN踩坑记录1 问题描述2 原因分析3 解决办法4 CAN配置参考代码 1 问题描述 GD32的CAN1无法进入接收中断&#xff0c;收不到数据。 注&#xff1a;MCU使用的是GD32E50x&#xff0c;其他型号不确定是否一样&#xff0c;本文只以GD32E50x举例说…

【Docker】gitea的ssh容器直通

本文首发于 ❄️慕雪的寒舍 1.跟着文档走 gitea的安装比较简单&#xff0c;直接使用官方文档中的docker-compose文件即可。如果想实现ssh容器直通&#xff0c;需要对这个docker-compose文件做一定修改。 如果你还没有安装docker&#xff0c;参考本站教程 linux安装docker&…

QT-贪吃蛇小游戏

QT-贪吃蛇小游戏 一、演示效果二、核心代码三、下载链接 一、演示效果 二、核心代码 #include "Food.h" #include <QTime> #include <time.h> #include "Snake.h"Food::Food(int foodSize):foodSize(foodSize) {coordinate.x -1;coordinate.…

多线程(4)——单例模式、阻塞队列、线程池、定时器

1. 多线程案例 1.1 单例模式 单例模式能保证某个类在程序中只存在唯一一份实例&#xff0c;不会创建出多个实例&#xff08;这一点在很多场景上都需要&#xff0c;比如 JDBC 中的 DataSource 实例就只需要一个 tip&#xff1a;设计模式就是编写代码过程中的 “软性约束”&am…

系统稳定性建设的深度剖析与未来展望

一、系统稳定性的重要意义 系统稳定性是系统正常运行的关键&#xff0c;其缺失会导致严重后果&#xff0c;如经济损失、用户流失等。 以在线学习平台为例&#xff0c;如果系统频繁出现卡顿、掉线等问题&#xff0c;影响用户的学习体验&#xff0c;导致用户流失&#xff0c;平…