【算法练习Day46】判断子序列不同的子序列

news2025/1/16 15:48:17

在这里插入图片描述

​📝个人主页:@Sherry的成长之路
🏠学习社区:Sherry的成长之路(个人社区)
📖专栏链接:练题
🎯长路漫漫浩浩,万事皆有期待

文章目录

  • 判断子序列
  • 不同的子序列
  • 总结:

判断子序列

392. 判断子序列 - 力扣(LeetCode)

判断子序列这道题目,和上一期的题解法几乎完全相同,只是递推公式有一点差别,但是要是完全用之前的代码也是可行的。

dp数组的含义:dp【i】【j】代表以i-1和j-1为结尾的相同子序列的长度。之前我们讲过,同时操作两个数组的时候,通常都是设置二维数组。设置为i-1和j-1表示的原因是,方便数组初始化。

递推公式:当两个字符相等时候

if(s【i-1】==t【j-1】)dp【i】【j】=dp【i-1】【j-1】+1;

因为dp数组含义的缘故,我们在判断时候对应的也是它的字符串的i-1位置和j-1位置,然后如果两字符此时对应相等,那么就是之前最大的相同子序列的长度+1。

else dp【i】【j】=dp【i】【j-1】;

如果两字符不相等,那么就让此时的对应的do等于 dp【i】【j-1】,这是因为我们是判断s是否为t的子序列,是删除t中某些元素看看能否构成s,已知此时两对应字符一定不相等,那此时dp就等于不考虑当前这个不等的字符时,所对应的最长相同子序列长度,这样做也相当于变相删除该多余字符。细心的读者也许已经发现了,与之前那期递推公式不同的是没有dp【i-1】【j】这一项了,之前的递推公式是它们两个取最大值,那是因为上一道题两个字符都可以删除若干数据,而最终保证结果是最长的公共部分就可以了,不一定给的哪个字符串更长,所以删哪一个也就不一定了,需要取最大值,而这道题我们是明确的判断s是否是t的子序列,那就是t长要删除某元素后看是否构成s。至于为什么我说用和之前一样的代码也能通过呢?因为我们不需要删除s的元素,也就是说加上它也不会影响代码的运行逻辑,但是建议还是要根据题目的需求写逻辑,而不是这类题都用一种固定的递推公式来写,这样多思考更有利于理解题的本质。

dp数组初始化:dp数组初始化就是全都是0没什么说的,由于dp定义的缘故,和递推公式的推理,都使得,应该被初始化为0。

遍历顺序:还是正常的从左到右从上到下。

class Solution {
public:
    bool isSubsequence(string s, string t) {
        vector<vector<int>>dp(s.size()+1,vector<int>(t.size()+1,0));
        for(int i=1;i<=s.size();i++){
            for(int j=1;j<=t.size();j++){
                if(s[i-1]==t[j-1])
                dp[i][j]=dp[i-1][j-1]+1;
                else dp[i][j]=dp[i][j-1];
            }
        }
        if(dp[s.size()][t.size()]==s.size())return true;
        return false;
    }
};

题整体来看如果,上一期的题目可以完全理解,那这道题就还是相对好做一些。

不同的子序列

115. 不同的子序列 - 力扣(LeetCode)

与上一道题的相比,这道题显得难了不少。也是困扰了我一段时间的题,感觉递推公式不是很好理解。

dp数组含义:这道题是判断在s里t出现的次数有多少次,这道题并不是说直观的就能看出s字符串里有几个t,不要曲解题目意思,它是说有多少种删除元素的方法,使得字符串由s转化为t,应该这样去理解。所以应该将数组含义定义为以i-1结尾的字符串s中包含有以j-1为结尾的t多少个。这个dp数组定义非常重要,要好好理解,才能够理解下面的递推公式。

递推公式:递推公式由两部分组成,和上一题一样,一个是要匹配的字符匹配,另一个字符不匹配,要模拟删除,而不同的是这道题里能够相匹配的字符递推公式,也分成两部分。

经过了一段时间的理解,我觉得这样想是能够帮助理解递推式的。

如果两字符不匹配,那么dp【i】【j】=dp【i-1】【j】因为我们上面说过dp数组的含义,表示的是前一个位置,当前一个位置不匹配字符,那么说明此时应该和同一列的上一行所代表的个数相等,为什么是这样呢?画dp数组推到图也就是打印,我们可以知道,s表示行t表示列时候,此时s对应的字符和t不等,那么就应该此处的数值继承下来上一次匹配成功的值,虽然它的同列上一行也可能不匹配,但是它的上一处也是继承的匹配成功的那个个数。我们是拿着s的字符一点点找t的字符去匹配,如果还是不明白,画一次会帮助理解。

那如果两字符相匹配呢?我们不仅仅要保留之前的匹配不成功的所继承下来的个数,还要加上此次匹配成功时候应该加上的个数,而上一次怕匹配成功是哪一个位置?是dp【i-1】【j-1】也就是我们要填写的位置的左上角,,它是匹配成功的时候所继承下来的含有个数,两者一做和就能得到答案了,所以递推公式是:

if(s【i-1】==t【j-1】)dp【i】【j】=dp【i-1】【j-1】+dp【i-1】【j】
else dp【i】【j】=dp【i-1】【j】

一开始怎么也想不明白为什么,要这么写,后来换一种思维去想,就能想得明白了。

dp数组初始化:初始化也是有一定讲究的,虽然我们是定义的i-1和j-1的位置,但此题并不意味着要全部初始化为0,因为我们是求得s里有几个t,当t是空字符串时候,每一个i-1下标都对应着一种方法得到t,那就是删除i-1之前的全部字符,得到一个空字符串。s是空字符串时候,无论怎么删都不可能得到一个t,而且其他的位置都会被递推公式所覆盖,所以除了第一行之外全都初始化为0就可以了,第一行初始化为1。

遍历顺序:根据递推公式得到,i,j位置由它的上面位置和左上位置推导,所以依旧是从上到下从左到右

class Solution {
public:
    int numDistinct(string s, string t) {
        vector<vector<unsigned long int>>dp(s.size()+1,vector<unsigned long int>(t.size()+1,0));
        dp[0][0]=1;
        for(int i=1;i<s.size();i++)dp[i][0]=1;
        for(int j=1;j<t.size();j++)dp[0][j]=0;
        for(int i=1;i<=s.size();i++){
            for(int j=1;j<=t.size();j++){
                if(s[i-1]==t[j-1])
                dp[i][j]=dp[i-1][j-1]+dp[i-1][j];//可以理解为dp二维数组即使本次匹配字符未成功
                //它也至少是上一次匹配的个数,因为dp数组的定义就是s字符串i-1位置下,对应的有几个t
                //本次即使未匹配成功,也不应该影响之前匹配成功保留下来的次数。
                //而如果匹配成功,那么就是两个字符匹配的成功个数加上本来有的个数
                else dp[i][j]=dp[i-1][j];
            }
        }
        return dp[s.size()][t.size()];
    }
};

总结:

今天我们完成了判断子序列、不同的子序列两道题,相关的思想需要多复习回顾。接下来,我们继续进行算法练习。希望我的文章和讲解能对大家的学习提供一些帮助。

当然,本文仍有许多不足之处,欢迎各位小伙伴们随时私信交流、批评指正!我们下期见~

在这里插入图片描述

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

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

相关文章

【C++笔记】二叉搜索树的模拟实现

【C笔记】二叉搜索树的模拟实现 一、二叉搜索树的概念二、二叉搜索树的模拟实现2.0、定义二叉树节点2.1、非递归接口实现2.1.1、插入2.1.2、查找2.1.3、删除 2.2、递归接口实现2.2.1、插入2.2.2、查找2.2.3、删除 三、升级为K-V模型 一、二叉搜索树的概念 二叉搜索树的概念&am…

grpc使用教程

准备 1&#xff0c;安装grpc go get -u google.golang.org/grpc2, 安装go语言protobuf生成器protoc-gen-go go get -u google.golang.org/protobuf/cmd/protoc-gen-go3, 通过下面连接&#xff0c;找到合适版本并安装protoc工具&#xff0c;如windows选择 protoc-3.19.5-win64.…

c++类对象内存模型(一)

C对象模型可以概括为以下2部分&#xff1a; 1. 语言中直接支持面向对象程序设计的部分&#xff0c;主要涉及如构造函数、析构函数、虚函数、继承&#xff08;单继承、多继承、虚继承&#xff09;、多态等等。 2. 对于各种支持的底层实现机制。在c语言中&#xff0c;“数据”和…

时间序列预测实战(十四)Transformer模型实现长期预测并可视化结果(附代码+数据集+原理介绍)

论文地址->Transformer官方论文地址 官方代码地址->暂时还没有找到有官方的Transformer用于时间序列预测的代码地址 个人修改地址-> Transformer模型下载地址CSDN免费 一、本文介绍 这篇文章给大家带来是Transformer在时间序列预测上的应用&#xff0c;这种模型最…

C++算法:包含三个字符串的最短字符串

涉及知识点 有序集合 字符串 题目 给你三个字符串 a &#xff0c;b 和 c &#xff0c; 你的任务是找到长度 最短 的字符串&#xff0c;且这三个字符串都是它的 子字符串 。 如果有多个这样的字符串&#xff0c;请你返回 字典序最小 的一个。 请你返回满足题目要求的字符串。…

2023年09月 Python(四级)真题解析#中国电子学会#全国青少年软件编程等级考试

Python等级考试(1~6级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 用枚举算法求解“100以内既能被3整除又能被4整除的元素”时,在下列数值范围内,算法执行效率最高的是?( ) A:1~101 B:4~100 C:12~100 D:12~96 答案:D 题目要求找出在 100…

面向对象特征【继承性】

文章目录 基本概念继承的语法继承性的细节方法的重写方法重写的要求 基本概念 继承性允许子类继承父类的属性和方法。在Java中&#xff0c;使用关键字extends来实现继承。例如&#xff1a; class Animal {void eat() {System.out.println("动物正在吃");} }class D…

绝了!现在制作电子期刊这么快而有效了?

​随着科技的进步&#xff0c;制作电子期刊已经变得越来越简单和高效。现在&#xff0c;您只需要一台电脑或手机&#xff0c;就可以快速制作出精美的电子期刊&#xff0c;而且还能有效地吸引读者的注意力。 但是如何快而有效的制作电子期刊呢&#xff1f; 1.首先打开FLBOOK在线…

2023第六届泰迪杯数据分析

第六届带队”指导“请私信本人&#xff0c;团队包含技能赛双一等&#xff0c;数学建模省一&#xff0c;泰迪杯挖掘国一&#xff0c;研究生队友。 去年一等作品可视化图如下&#xff0c;私信获取源码

[直播自学]-[汇川easy320]搞起来(4)看文档 查找设备(续)

2023.11.12 周六 19&#xff1a;05 补充一下关于以太网查找设备&#xff0c;如果设置如下&#xff1a; 然后点击测试&#xff1a; 点击ping&#xff1a;

Windows下Oracle安装和卸载

Windows下Oracle安装和卸载 1、Windows下安装Oracle 安装的版本&#xff1a;win32_11gR2_database。 解压之后双击setup.exe程序。 点击是。 配置安全更新&#xff0c;去掉复选框&#xff0c;点下一步。 提示未指定电子邮件地址&#xff0c;点是跳过。 配置安装选项&#xf…

Java继承和多态(1)

&#x1f435;本主题将分为篇文章&#xff0c;本篇文章将主要对继承进行讲解 一、介绍继承 1.1 什么是继承 假如有两个类&#xff1a;A类和B类&#xff0c;A类在保持原有成员变量和方法的基础上可以使用B类的成员变量和方法&#xff0c;此时就称A类继承了B类&#xff0c;A类为…

4.1每日一题(多元函数微分:偏导数判定(链导法、定义、先代后求))

链导法&#xff1a;如果每一层复合都可导&#xff08;即在判断点要可导&#xff09;&#xff0c;则复合函数一定可导&#xff1b;如果中间层有不可导&#xff0c;就不方便使用链导法了&#xff08;在&#xff08;0&#xff0c;0&#xff09;点不可导) 当链导法不好用时应该直接…

微信支付平台C#SDK_微信支付.net SDK

一、微信支付平台C# SDK V3 https://github.com/fudiwei/DotNetCore.SKIT.FlurlHttpClient.Wechat/blob/main/docs/WechatTenpayV3 接口对应整理&#xff1a; https://github.com/fudiwei/DotNetCore.SKIT.FlurlHttpClient.Wechat/blob/main/docs/WechatTenpayV3/Basic_Mod…

vue3响应式api

响应式api——compositon api setup&#xff1a; 不要再想this问题执行是在beforeCreated之前 beforeCreated&#xff1a;也就是创建了一个实例 created&#xff1a;挂载了数据 通过形参props接收&#xff0c;只读 以后所有代码都写到setup中 判断是否只读&#xff1a;isReadon…

大数据可视化数据大屏可视化模板【可视化项目案例-05】

🎉🎊🎉 你的技术旅程将在这里启航! 🚀🚀 本文选自专栏:可视化技术专栏100例 可视化技术专栏100例,包括但不限于大屏可视化、图表可视化等等。订阅专栏用户在文章底部可下载对应案例源码以供大家深入的学习研究。 🎓 每一个案例都会提供完整代码和详细的讲解,不…

数据结构:AVLTree的插入和删除的实现

个人主页 &#xff1a; 个人主页 个人专栏 &#xff1a; 《数据结构》 《C语言》《C》 文章目录 前言一、AVLTree二、AVLTree的插入插入新增节点调整平衡因子旋转左单旋(新增节点位于较高右子树的右侧)右单旋(新增节点位于较高左子树的左侧)右左双旋(新增节点在较高右子树的左子…

多数据源切换

多数据源切换 1.jdbcTemplate2.使用切面3.mybatis层次的多数据源4.spring的dynamic自动注入 项目中经常会有多个数据源&#xff0c;那么如何处理呢 有4种方法 准备&#xff1a; 创建两个数据库 CREATE SCHEMA test DEFAULT CHARACTER SET utf8mb4 ; CREATE SCHEMA school DEFA…

pandas笔记:读写excel

1 读excel read_excel函数能够读取的格式包含&#xff1a;xls, xlsx, xlsm, xlsb, odf, ods 和 odt 文件扩展名。 支持读取单一sheet或几个sheet。 1.0 使用的数据 1.1 主要使用方法 pandas.read_excel(io, sheet_name0, header0, namesNone, index_colNone, usecolsNon…

Nginx:如何实现一个域名访问多个项目

1. 背景介绍 最近在多个项目部署中遇到这样一个问题&#xff0c;一个域名如何实现多个项目的访问。因为不想自己单独去申请域名证书和域名配置&#xff0c;便想到了这个方案&#xff0c;结合Nginx的location功能实现了自己的需求&#xff0c;便记录下来。示例中是以项目演示&a…