力扣练习题(2024/4/18)

1不相交的线

在两条独立的水平线上按给定的顺序写下 nums1 和 nums2 中的整数。

现在,可以绘制一些连接两个数字 nums1[i] 和 nums2[j] 的直线,这些直线需要同时满足:

  •  nums1[i] == nums2[j]
  • 且绘制的直线不与任何其他连线(非水平线)相交。

请注意,连线即使在端点也不能相交:每个数字只能属于一条连线。

以这种方法绘制线条,并返回可以绘制的最大连线数。

示例 1:

输入:nums1 = [1,4,2], nums2 = [1,2,4]
输出:2
解释:可以画出两条不交叉的线,如上图所示。 
但无法画出第三条不相交的直线,因为从 nums1[1]=4 到 nums2[2]=4 的直线将与从 nums1[2]=2 到 nums2[1]=2 的直线相交。

示例 2:

输入:nums1 = [2,5,1,2,5], nums2 = [10,5,2,1,5,2]
输出:3

示例 3:

输入:nums1 = [1,3,7,1,7,5], nums2 = [1,9,2,5,1]
输出:2

提示:

  • 1 <= nums1.length, nums2.length <= 500
  • 1 <= nums1[i], nums2[j] <= 2000

思路:

绘制一些连接两个数字 A[i] 和 B[j] 的直线,只要 A[i] == B[j],且直线不能相交!

直线不能相交,这就是说明在字符串A中 找到一个与字符串B相同的子序列,且这个子序列不能改变相对顺序,只要相对顺序不改变,链接相同数字的直线就不会相交。拿示例nums1 = [1,4,2], nums2 = [1,2,4]为例,相交情况如图:

其实也就是说nums1和nums2的最长公共子序列是[1,4],长度为2。 这个公共子序列指的是相对顺序不变(即数字4在字符串A中数字1的后面,那么数字4也应该在字符串B数字1的后面)

这么分析完之后,大家可以发现:本题说是求绘制的最大连线数,其实就是求两个字符串的最长公共子序列的长度!

首先,我们可以定义一个二维数组 dp,其中 dp[i][j] 表示 nums1 的前 i 个数字和 nums2 的前 j 个数字所能组成的最大不相交连线数。

然后,我们可以考虑状态转移方程。当 nums1[i-1] == nums2[j-1] 时,说明 nums1 的第 i 个数字和 nums2 的第 j 个数字可以连线,此时最大连线数为 dp[i-1][j-1] + 1。否则,说明 nums1[i-1] 和 nums2[j-1] 不能连线,那么最大连线数为 max(dp[i-1][j], dp[i][j-1]),即不考虑当前这对数字时的最大连线数。

最后,我们遍历 nums1 和 nums2 的所有组合情况,计算得到  dp[nums1.size()][nums2.size()]即为所求的结果

代码:

class Solution {
public:int maxUncrossedLines(vector<int>& nums1, vector<int>& nums2) {// 创建二维动态规划数组,大小为 nums1.size()+1 行,nums2.size()+1 列,初始化为0vector<vector<int>> dp(nums1.size() + 1, vector<int>(nums2.size() + 1, 0));// 遍历 nums1 和 nums2 的所有组合情况for (int i = 1; i <= nums1.size(); i++) {for (int j = 1; j <= nums2.size(); j++) {if (nums1[i - 1] == nums2[j - 1]) {// 当 nums1[i-1] 和 nums2[j-1] 相等时,更新 dp[i][j] 为左上角值加一dp[i][j] = dp[i - 1][j - 1] + 1;} else {// 否则,更新 dp[i][j] 为上方和左方值的较大者dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);}}}// 返回最终结果,即 dp[nums1.size()][nums2.size()]return dp[nums1.size()][nums2.size()];}
};

2 最大子数组和

给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

子数组

是数组中的一个连续部分。

示例 1:

输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。

示例 2:

输入:nums = [1]
输出:1

示例 3:

输入:nums = [5,4,-1,7,8]
输出:23

提示:

  • 1 <= nums.length <= 105
  • -104 <= nums[i] <= 104

思路:

首先,我们定义一个动态规划数组 dp,其中 dp[i] 表示以 nums[i] 结尾的子数组的最大和。

然后,我们考虑状态转移方程。对于数组中的第 i 个元素 nums[i],我们有两种选择:要么将其加入前面的子数组,形成一个新的子数组;要么将其作为新的起点,重新开始一个新的子数组。因此

1 dp[i]只有两个方向可以推出来:
  • dp[i - 1] + nums[i],即:nums[i]加入当前连续子序列和
  • nums[i],即:从头开始计算当前连续子序列和

一定是取最大的,所以dp[i] = max(dp[i - 1] + nums[i], nums[i]);

这个方程的含义是,要么当前元素加上前一个元素的子数组和构成了一个更大的子数组,要么当前元素本身构成了一个新的子数组。我们选择其中较大的那个作为 dp[i] 的值。

2 dp数组如何初始化  

从递推公式可以看出来dp[i]是依赖于dp[i - 1]的状态,dp[0]就是递推公式的基础。

dp[0]应该是多少呢?

根据dp[i]的定义,很明显dp[0]应为nums[0]即dp[0] = nums[0]。

3  确定遍历顺序

递推公式中dp[i]依赖于dp[i - 1]的状态,需要从前向后遍历。

代码: 

class Solution {
public:int maxSubArray(vector<int>& nums) {// 如果数组为空,则直接返回0if (nums.size() == 0) return 0;// 创建动态规划数组,大小为 nums.size(),用于存储最大子数组和vector<int> dp(nums.size());dp[0] = nums[0]; // 初始化动态规划数组的第一个元素为数组的第一个元素int result = dp[0]; // 用于存储最大子数组和的变量,初始化为数组的第一个元素// 遍历数组,计算最大子数组和for (int i = 1; i < nums.size(); i++) {// 状态转移公式:当前元素加上前一个元素的动态规划值,或者当前元素本身,取两者中的较大值dp[i] = max(dp[i - 1] + nums[i], nums[i]);// 更新最大子数组和的值if (dp[i] > result) result = dp[i];}// 返回最终的最大子数组和return result;}
};

3 判断子序列

给定字符串 s 和 t ,判断 s 是否为 t 的子序列。

字符串的一个子序列是原始字符串删除一些(也可以不删除)字符而不改变剩余字符相对位置形成的新字符串。(例如,"ace""abcde"的一个子序列,而"aec"不是)。

进阶:

如果有大量输入的 S,称作 S1, S2, ... , Sk 其中 k >= 10亿,你需要依次检查它们是否为 T 的子序列。在这种情况下,你会怎样改变代码?

致谢:

特别感谢 @pbrother 添加此问题并且创建所有测试用例。

示例 1:

输入:s = "abc", t = "ahbgdc"
输出:true

示例 2:

输入:s = "axc", t = "ahbgdc"
输出:false

 双指针思路:

  1. 定义两个指针 i 和 j,分别用于遍历两个序列 nums1 和 nums2
  2. 初始时,i 和 j 都指向两个序列的起始位置。
  3. 在遍历过程中,比较 nums1[i] 和 nums2[j] 是否相等。
  4. 如果相等,则将 j 移动到 nums2 的下一个位置。
  5. 无论是否相等,都将 i 移动到 nums1 的下一个位置。
  6. 当遍历完 nums1 或者 nums2 中的任意一个时,停止遍历。
  7. 如果 j 已经达到 nums2 的末尾,则说明 nums2 是 nums1 的子序列,返回 true
  8. 如果遍历完 nums1 后,j 还没有到达 nums2 的末尾,则说明 nums2 不是 nums1 的子序列,返回 false

代码:

#include <vector>class Solution {
public:bool isSubsequence(std::vector<int>& nums1, std::vector<int>& nums2) {int i = 0, j = 0;int n = nums1.size();int m = nums2.size();// 遍历 nums1while (i < n && j < m) {// 如果 nums1[i] 与 nums2[j] 相等if (nums1[i] == nums2[j]) {// 移动 nums2 的指针j++;}// 无论是否匹配,nums1 的指针都移动到下一个位置i++;}// 如果 nums2 的指针到达末尾,则 nums2 是 nums1 的子序列return j == m;}
};

动态规划思路:

确定dp数组(dp table)以及下标的含义

dp[i][j] 表示以下标i-1为结尾的字符串s,和以下标j-1为结尾的字符串t,相同子序列的长度为dp[i][j]

注意这里是判断s是否为t的子序列。即t的长度是大于等于s的。

然后,我们初始化动态规划数组 dp 的大小为 (s.size() + 1) * (t.size() + 1),并将所有元素初始化为 0。接着,我们遍历字符串 s 和 t 的所有字符,外层循环遍历 s 的每个字符,内层循环遍历 t 的每个字符。

在遍历过程中,如果 s 的第 i 个字符等于 t 的第 j 个字符,则说明找到了一个相同的字符,此时 dp[i][j] 的值为 dp[i - 1][j - 1] + 1,表示将该字符加入子序列中。如果 s 的第 i 个字符不等于 t 的第 j 个字符,则说明当前字符不能加入子序列,此时 dp[i][j] 的值为 dp[i][j - 1],表示不将该字符加入子序列中。

具体推出递推公式思路如下:

  • if (s[i - 1] == t[j - 1])
    • t中找到了一个字符在s中也出现了
  • if (s[i - 1] != t[j - 1])
    • 相当于t要删除元素,继续匹配
  • if (s[i - 1] == t[j - 1]),那么dp[i][j] = dp[i - 1][j - 1] + 1;,因为找到了一个相同的字符,相同子序列长度自然要在dp[i-1][j-1]的基础上加1
  • if (s[i - 1] != t[j - 1]),此时相当于t要删除元素,t如果把当前元素t[j - 1]删除,那么dp[i][j] 的数值就是 看s[i - 1]与 t[j - 2]的比较结果了,即:dp[i][j] = dp[i][j - 1];

遍历完成后,如果 dp[s.size()][t.size()] 的值等于 s 的长度,则说明 s 是 t 的子序列,返回 true;否则返回 false

代码:

#include <vector>
#include <string>class Solution {
public:// 判断字符串 s 是否是字符串 t 的子序列bool isSubsequence(string s, string t) {// 创建二维动态规划数组,dp[i][j] 表示 s 的前 i 个字符是否是 t 的前 j 个字符的子序列vector<vector<int>> dp(s.size() + 1, vector<int>(t.size() + 1, 0));// 遍历 s 和 t 的所有字符for (int i = 1; i <= s.size(); i++) {for (int j = 1; j <= t.size(); j++) {// 如果 s 的第 i 个字符等于 t 的第 j 个字符if (s[i - 1] == t[j - 1]) {// 则 s 的前 i 个字符作为 t 的前 j 个字符的子序列的长度加一dp[i][j] = dp[i - 1][j - 1] + 1;} else {// 否则,s 的前 i 个字符作为 t 的前 j 个字符的子序列的长度与之前的相同dp[i][j] = dp[i][j - 1];}}}// 如果 s 的所有字符都是 t 的子序列,则返回 true,否则返回 falseif (dp[s.size()][t.size()] == s.size()) {return true;}return false;}
};

4不同的子序列

给你两个字符串 s 和 t ,统计并返回在 s 的 子序列 中 t 出现的个数,结果需要对 109 + 7 取模。

示例 1:

输入:s = "rabbbit", t = "rabbit"
输出3
解释:
如下所示, 有 3 种可以从 s 中得到 "rabbit" 的方案rabbbit
rabbbit
rabbbit

示例 2:

输入:s = "babgbag", t = "bag"
输出5
解释:
如下所示, 有 5 种可以从 s 中得到 "bag" 的方案babgbag
babgbag
babgbag
babgbag
babgbag

提示:

  • 1 <= s.length, t.length <= 1000
  • s 和 t 由英文字母组成

动态规划思路:

  1. 定义 dp[i][j] 为以 s 的第 i-1 个字符结尾的子序列中出现以 t 的第 j-1 个字符结尾的子序列的个数。

  2. 初始状态:

    • 当 t 为空字符串时,s 的任何子串都可以与之匹配,因此 dp[i][0] = 1
    • 当 s 为空字符串时,无论 t 如何都无法匹配,因此 dp[0][j] = 0。这一步其实在代码中被默认初始化为 0,因此省略了这一步的实际操作。
  3. 状态转移方程:

    • 当 s[i - 1] == t[j - 1] 时,即 s 的第 i-1 个字符与 t 的第 j-1 个字符相等,此时 dp[i][j] 可以由两部分得到:
      1. 不考虑 s[i-1] 和 t[j-1] 的情况,即 dp[i-1][j]
      2. 考虑 s[i-1] 和 t[j-1] 的情况,即 dp[i-1][j-1]。因为此时 s 的第 i-1 个字符与 t 的第 j-1 个字符匹配,所以这种情况下 dp[i][j] 的数量应该加上以 s 的第 i-2 个字符结尾的子序列中以 t 的第 j-2 个字符结尾的子序列的数量。
      3. 还要考虑 不用s[i - 1]来匹配,都相同了指定要匹配啊

        例如: s:bagg 和 t:bag ,s[3] 和 t[2]是相同的,但是字符串s也可以不用s[3]来匹配,即用s[0]s[1]s[2]组成的bag。

        当然也可以用s[3]来匹配,即:s[0]s[1]s[3]组成的bag。

        所以当s[i - 1] 与 t[j - 1]相等时,dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];

    • 当 s[i - 1] != t[j - 1] 时,即 s 的第 i-1 个字符与 t 的第 j-1 个字符不相等,此时 dp[i][j] 只能由 dp[i-1][j] 得到。

      当s[i - 1] 与 t[j - 1]不相等时,dp[i][j]只有一部分组成,不用s[i - 1]来匹配(就是模拟在s中删除这个元素),即:dp[i - 1][j]

      所以递推公式为:dp[i][j] = dp[i - 1][j];

  4. 最后返回 dp[s.size()][t.size()],即 s 和 t 的完全匹配的子序列数量。

代码:

class Solution {
public:int numDistinct(string s, string t) {// 创建一个二维向量 dp,用于存储状态vector<vector<uint64_t>> dp(s.size() + 1, vector<uint64_t>(t.size() + 1));// 初始化 dp,当 t 为空字符串时,s 的子串都可以与 t 匹配,所以为 1for (int i = 0; i < s.size(); i++) dp[i][0] = 1;// 当 s 为空字符串时,无论 t 如何都不能匹配,所以为 0(默认初始化时已为 0,此处可省略)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++) {// 如果 s 的第 i 个字符等于 t 的第 j 个字符if (s[i - 1] == t[j - 1]) {// 当前状态可以由前一个状态和前一个状态减去 s 的第 i 个字符 的状态得到dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];} else {// 当前状态可以由前一个状态得到dp[i][j] = dp[i - 1][j];}}}// 返回 s 和 t 的匹配数return dp[s.size()][t.size()];}
};

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

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

相关文章

省级客运、货运量及周转量数据(1990-2022年)

1、数据介绍 客运量和货运量是衡量交通运输行业发展状况的重要指标&#xff0c;可以反映一个地区或国家的经济发展水平和人民生活水平。而周转量则是反映运输行业效率的指标&#xff0c;即货物或旅客被运输的总距离。 省级客运、货运量及周转量是衡量一个地区交通运输行业发展…

Excel如何计算时间差

HOUR(B1-A1)&"小时 "&MINUTE(B1-A1)&"分钟 "&SECOND(B1-A1)&"秒"

手拉手安装Kafka2.13发送和消费消息

Kafka是一种高吞吐量的分布式发布订阅消息系统&#xff0c;它可以处理消费者在网站中的所有动作流数据。 Kafka启动方式有Zookeeper和Kraft&#xff0c;两种方式只能选择其中一种启动&#xff0c;不能同时使用。 Kafka下载https://downloads.apache.org/kafka/3.7.0/kafka_2.…

C语言 选择控制结构(1) 了解选择结构 关系运算符讲解 基本逻辑判断演示

接下来 我们来说 选择控制结构 在生活中 我们也有很多需要分支结构的例子 比如: 计算两个整数的最大值 计算n个数的最大值&#xff0c;最小值 判断三角形三边能否构成三角形? 判断某年是否是闰年? 判断输入的英文字母是大写还是小写? 我们在程序开发中 需要根据某种条件 进…

Mysql 、Redis 数据双写一致性 更新策略与应用

零、important point 1. 缓存双写一致性问题 2. java实现逻辑&#xff08;对于 QPS < 1000 可以使用&#xff09; public class UserService {public static final String CACHE_KEY_USER "user:";Resourceprivate UserMapper userMapper;Resourceprivate Re…

学生选课及成绩查询管理系统的设计与开发C#(winform + sqlserver)

源码来自网络 技术栈&#xff1a; C#的窗体程序开发 本系统未采用C#实现MDI——多文档窗口&#xff0c;因为考虑到C#的该技术与java类似&#xff0c;而暑期java实训时&#xff0c;曾用过类似的方法做过停车场管理系统&#xff0c;所以想为这次的系统注入一点新鲜的血液&#x…

HTML随机点名程序

案例要求 1.点击点名按钮&#xff0c;名字界面随机显示&#xff0c;按钮文字由点名变为停止 2.再次点击点名按钮&#xff0c;显示当前被点名学生姓名&#xff0c;按钮文字由停止变为点名 案例源码 <!DOCTYPE html> <html lang"en"> <head> <m…

虚拟机中安装的CentOS7的桌面右上角没有网络图标解决方案

问题描述 今天在打开CentOS7后&#xff0c;发现右上角的网络图标不见了&#xff0c;然后命令行访问百度也不通。然后上网查了一些解决方法。 原因分析及解决方案&#xff1a; 上网查了许多解决方法&#xff0c;其中一种成功解决了我的问题&#xff1b;我的是配置文件的问题。…

1-内核开发环境ubuntu+virtualbox+mobaXterm搭建

内核开发环境 ubuntuvirtualboxmobaXterm搭建 目录 内核开发环境 ubuntuvirtualboxmobaXterm搭建 1.virtualbox 安装 2.ubuntu 安装 3.网络设置 4.虚拟机安装ssh 服务&#xff0c;更新ubuntu 源安装基本软件 5.mobaXterm 个人免费版本安装 6.总结 本课程教程从0-1开始教…

Ansys Zemax|如何使用坐标返回功能恢复原坐标系

附件下载 联系工作人员获取附件 在OpticStudio的序列模式下&#xff0c;坐标间断面&#xff08;CB&#xff0c;Coordinate Break&#xff09;用于根据当前系统定义新的坐标系。本文将介绍如何在OpticStudio中使用坐标返回功能。 坐标返回求解可以方便地自动恢复到所需表面的…

LangChain4j

文章目录 关于 LangChain4j特性2 levels of abstractionLibrary StructureTutorials (User Guide)Integrations and Models免责声明 Highlights定义由LLM提供支持的声明性 AI Services&#xff1a;使用 LLM 分类从非结构数据中提取结构化信息 Getting started兼容性 支持的 LLM…

jdk版本升级,导致项目编译报错:java: 程序包javax.annotation不存在

1.问题异常&#xff1a; java: 程序包javax.annotation不存在。 2.发生背景&#xff1a; 项目使用jdk版本做了升级&#xff0c;升级到了jdk19。结果切换了以后项目编译就报错了&#xff0c;报错信息就是上述的&#xff1a;java: 程序包javax.annotation不存在。 3.报错原因&…

数据库服务类--Redis--未授权访问终端Getshell

免责声明:本文仅做技术交流与学习. 目录 前提条件: windows上开启redis服务: Linux上创建&开启redis服务: 操作: 1-连接靶机redis 2-写入webshell 3-访问后门 redis--->webshell Redis未授权访问漏洞复现与利用 - 知乎 (zhihu.com) 前提条件: 端口开放(6379) 目录…

vscode自动生成项目目录结构

目录结构如下&#xff1a; 生成步骤如下&#xff1a; vscode安装插件&#xff0c;project-tree安装之后按ctrlshiftp&#xff0c;并输入Project Tree回车点击要生成目录的项目&#xff0c;回车将项目目录生成并存储到README.md中

vue3中web前端JS动画案例(四)侧边栏横幅效果-右下角广告-淘宝案例

myJsAnimation.js, 这里使用了上次封装的动画方法&#xff0c;并进行了改造 /*** 动画的函数* dom 当前对象* JSON 传入元素对象的属性 {"width": 300, "opacity": 50}* * -------------------- 多物体运动&#xff0c;同时运动 ---传入JSON-------------*…

Nodejs 第六十七章(OpenAI)

OpenAI OpenAI是一个人工智能研究实验室和技术公司&#xff0c;致力于推动人工智能的发展和应用 OpenAI最著名的项目之一是GPT&#xff08;Generative Pre-trained Transformer&#xff09;系列模型&#xff0c;其中包括了GPT-3&#xff0c;它是迄今为止最大规模的语言模型之…

Python | Leetcode Python题解之第43题字符串相乘

题目&#xff1a; 题解&#xff1a; class Solution:def multiply(self, num1: str, num2: str) -> str:if num1 "0" or num2 "0":return "0"m, n len(num1), len(num2)ansArr [0] * (m n)for i in range(m - 1, -1, -1):x int(num1[i…

BI建设案例:FineBI大数据分析平台助力工程机械行业降本增效

工程机械行业作为国民经济的重要支柱&#xff0c;产品多样化、应用广泛&#xff0c;市场集中度高。其上游涉及原材料和核心零部件&#xff0c;下游则与房地产、基建工程和采矿等行业紧密相连。 如今&#xff0c;中国已崛起为全球工程机械制造大国&#xff0c;各类机械产品产量…

Macs Fan Control Pro for Mac:全面优化Mac风扇控制软件

Macs Fan Control Pro for Mac是一款专为苹果电脑用户设计的风扇控制软件&#xff0c;旨在通过精确的风扇速度调节&#xff0c;全面优化Mac的散热性能&#xff0c;确保系统始终运行在最佳状态。 Macs Fan Control Pro for Mac中文版下载 该软件具备实时监控功能&#xff0c;能够…

【CSS】grid 布局一行自动填充,每行最大限定px

<div class"model-plat-content"><div class"mode-card" v-for"i in 30"></div></div>.model-plat-content {display: grid;// 解释&#xff1a; repeat(auto-fit, minmax(250px, 1fr)) 自动填充&#xff0c;每行最大25…