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

news2024/11/17 17:24:42

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 列,初始化为0
        vector<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) {
        // 如果数组为空,则直接返回0
        if (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();
        
        // 遍历 nums1
        while (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,否则返回 false
        if (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 匹配,所以为 1
        for (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.coloradmin.cn/o/1610801.html

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

相关文章

kaggle 房价预测 得分0.53492

流程 导入需要的包引入文件,查看内容数据处理调用模型准备训练输出结果 导入需要的包 import pandas as pd import numpy as np import matplotlib.pyplot as plt import seaborn as sns from sklearn.model_selection import train_test_split from sklearn.linear_model i…

Pandas介绍与Series创建

1.Pandas介绍 Pandas 是基于 NumPy 的一种工具&#xff0c;该工具是为解决数据分析任务而创建的&#xff0c;Pandas 提供了大量能使我们快速便捷地处理数据的功能 Pandas 与出色的 Jupyter 工具包和其他库相结合&#xff0c;Python 中用于进行数据分析的环境在性能、生产率和协…

【介绍下WebStorm开发插件】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

稀碎从零算法笔记Day53-LeetCode:不同路径 II

稀碎系列有点更不动(更多是自己懈怠了) 题型&#xff1a;矩阵、模拟 链接&#xff1a;63. 不同路径 II - 力扣&#xff08;LeetCode&#xff09; 来源&#xff1a;LeetCode 题目描述 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &…

Emerald AI 2024

使用易于使用的编辑器和大量内置功能,快速创建高质量的人工智能。 Emerald AI 2024是一个完全重写和重新设计的通用人工智能框架,适用于各种人工智能和游戏类型。它的多组件设计使开发人员能够灵活地只使用他们需要的功能,并允许有组织和可管理的工作流程。Emerald AI经过了…

【介绍下LeetCode的使用方法】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

OpenHarmony其他工具类—libharu [GN编译]

简介 libharu主要用于生成 PDF格式文件。 下载安装 直接在OpenHarmony-SIG仓中搜索libharu并下载。 使用说明 以OpenHarmony 3.1 Beta的rk3568版本为例 库代码存放路径&#xff1a;./third_party/libharu 修改添加依赖的编译脚本&#xff0c;路径&#xff1a;/developtools…

OpenHarmony南向开发案例:【智能中控屏】

样例简介 本Demo是基于Hi3516开发板&#xff0c;使用开源OpenHarmony开发的应用。通过控制面板可以控制同一局域网内的空调&#xff0c;窗帘&#xff0c;灯等智能家居设备。 当前支持的配套L0设备只有[智能灯]&#xff0c;如需添加新的设备。 应用运行效果图&#xff1a; 样…

nginx--Nginx转发真实的IP

Nginx转发真实的IP 前言给nginx.conf 设置proxy_set_headerjava 程序里获取 前言 在使用nginx的时候可能会遇到判断是不是本机在做操作&#xff0c;这样的话web端我们是可以通过ip和端口进行远程连接的这样的话我们就需要从后端获取到真实ip来判断是不是指定的机器了&#xff…

PS-ZB转座子分析流程2-重新分析并总结

数据处理 数据质控 随机挑出九个序列进行比对&#xff0c;结果如下&#xff1a; 所有序列前面的部分序列均完全相同&#xff0c;怀疑是插入的转座子序列&#xff0c;再随机挑选9个序列进行比对&#xff0c;结果如下&#xff1a; 结果相同&#xff0c;使用cutadapt将该段序列修…

【C语言】贪吃蛇项目(2)- 实现代码详解

文章目录 前言一、游戏开始界面设计首先 - 打印环境界面其次 - 游戏地图、蛇身及食物的设计1、地图2、蛇身设置及打印3、食物 二、游戏运行环节蛇的上下左右移动等功能蛇的移动 三、结束游戏代码 前言 在笔者的前一篇博客中详细记载了贪吃蛇项目所需的一些必备知识以及我们进行…

【飞桨AI实战】人体姿态估计:零基础入门,从模型训练到应用开发

前言 本次分享将带领大家从 0 到 1 完成一个人体姿态估计任务&#xff0c;覆盖数据准备、模型训练、推理部署和应用开发的全流程&#xff0c;项目将采用以PaddlePaddle为核心的飞桨深度学习框架进行开发&#xff0c;并总结开发过程中踩过的一些坑&#xff0c;希望能为有类似项…

模电期末复习(二)放大电路的基本原理和分析方法

放大电路的基本原理和分析方法 2.1 放大的概念2.2 放大电路的主要技术指标2.3 单管共发射极放大电路2.3.1 单管共发射极放大电路的组成2.3.2 单管共射放大电路的工作原理 2.4 放大电路的基本分析方法2.4.1 直流通路与交流通路2.4.2 静态工作点的近似估算2.4.3 图解法&#xff…

第23天:安全开发-PHP应用后台模块SessionCookieToken身份验证唯一性

第二十三天 一、PHP后台身份验证模块实现 二、Cookie&Session技术&差异 1.生成cookie的原理图过程&#xff1a;见上图 客户端向服务器发送HTTP请求。服务器检查请求头中是否包含cookie信息。如果请求头中包含cookie信息&#xff0c;则服务器使用该cookie来识别客户端…

C++奇迹之旅:构造函数和析构函数

文章目录 &#x1f4dd;类的6个默认成员函数&#x1f320; 构造函数&#x1f309; 概念&#x1f309;特性&#x1f309;三种默认构造函数 &#x1f320; 析构函数&#x1f320; 特性&#x1f6a9;总结 &#x1f4dd;类的6个默认成员函数 如果一个类中什么成员都没有&#xff0…

OpenHarmony其他工具类—lua

简介 Lua是一种功能强大、高效、轻量级、可嵌入的脚本语言。 支持过程编程、面向对象编程、函数编程、数据驱动编程和数据描述。 下载安装 直接在OpenHarmony-SIG仓中搜索lua并下载。 使用说明 以OpenHarmony 3.1 Beta的rk3568版本为例 将下载的lua库代码存在以下路径&#…

javase__进阶 day13stream流和方法引用

1.不可变集合 1.1 什么是不可变集合 ​ 是一个长度不可变&#xff0c;内容也无法修改的集合 1.2 使用场景 ​ 如果某个数据不能被修改&#xff0c;把它防御性地拷贝到不可变集合中是个很好的实践。 ​ 当集合对象被不可信的库调用时&#xff0c;不可变形式是安全的。 简单…

von Mises-Fisher Distribution (代码解析)

torch.distribution 中包含了很多概率分布的实现&#xff0c;本文首先通过均匀分布来说明 Distribution 的具体用法, 然后再解释 von Mises-Fisher 分布的实现, 其公式推导见 von Mises-Fisher Distribution. 1. torch.distribution.Distribution 以下是 Uniform 的源码: cl…

黑灰产行业简介

参考&#xff1a;2021年黑灰产行业研究及趋势洞察报告 1. 有哪些场景面临大量黑灰产攻击&#xff1f; 1.营销活动场景 -- 该场景最为猖獗 1. 抹机及接码注册&#xff1a;黑灰产会使用抹机工具修改设备参数伪装成一台新设备&#xff0c;再配合联系卡商进行手机号接码&#xf…

面试(05)————Redis篇

目录 一、项目中哪些地方使用了redis 问题一&#xff1a;发生了缓存穿透该怎么解决&#xff1f; 方案一&#xff1a;缓存空数据 方案二&#xff1a;布隆过滤器 模拟面试 问题二&#xff1a; 发生了缓存击穿该怎么解决&#xff1f; 方案一&#xff1a;互斥锁 方案二&#xff…