Day46 | 动态规划 :线性DP 最长递增子序列最长连续递增子序列

Day46 | 动态规划 :线性DP 最长递增子序列&&最长连续递增子序列

动态规划应该如何学习?-CSDN博客

本次题解参考自灵神的做法,大家也多多支持灵神的题解

最长递增子序列【基础算法精讲 20】_哔哩哔哩_bilibili

动态规划学习:

1.思考回溯法(深度优先遍历)怎么写

注意要画树形结构图

2.转成记忆化搜索

看哪些地方是重复计算的,怎么用记忆化搜索给顶替掉这些重复计算

3.把记忆化搜索翻译成动态规划

基本就是1:1转换

文章目录

  • Day46 | 动态规划 :线性DP 最长递增子序列&&最长连续递增子序列
    • 300.最长递增子序列
      • 思路分析(子问题):
      • 1.回溯 DFS
      • 2.记忆化搜索
      • 3.1:1翻译为动态规划
    • 674.最长连续递增序列
      • 思路:
      • 1.回溯法
      • 2.记忆化搜索
      • 3.动态规划

300.最长递增子序列

300. 最长递增子序列 - 力扣(LeetCode)

思路分析(子问题):

dfs(i)表示以nums[i]结尾的最长递增子序列的长度

我们往前找,只有找到比nums[i]小的nums[j]才往下递归,否则不进行递归

我们一次dfs(i)可以找出以nums[i]结尾的最长递增子序列的长度

dfs(i)=max(dfs(j))+1 {0<j<i}

加1加的是nums[i]本身

举个例子:

我们的i如果是5的话,nums[i]=7,枚举比它小的,即i=4,3,2 nums[i]=3,5,2

输入:nums = [10,9,2,5,3,7,101,18]
输出:4
dfs(5)=max(dfs(4),dfs(3),dfs(2))+1

这样即可以把最长的情况枚举出来,还可以避免遗漏,因为每一次dfs我们都只找以nums[i]结尾的最长的递增子序列

如果以7结尾的话,那就是2,3,7

如果我们要再找以18结尾的

那就是

dfs(7)=max(dfs(5),dfs(4),dfs(3),dfs(2),dfs(1),dfs(0))+1

如果大家没有听太懂的话建议看看灵神的视频(知道那个意思,但是我也不知道该怎么讲)

1.回溯 DFS

1.返回值和参数

i是数组下标,我们最长递增子序列的最后一个数就是nums[i]

dfs(i)表示以nums[i]结尾的最长递增子序列的长度

int dfs(int i,vector<int>& nums)

2.终止条件

终止条件就是i<0,即子序列里面没有数字了

i<0我们的程序本身也不会执行,所以不需要终止条件

3.本层逻辑

我们以nums[i]结尾的最长递增子序列的长度不需要看i以后的数,所以每次从0-i进行枚举,如果比nums[i]才会继续往下选

我们一次dfs(i)算出来的是以nums[i]为结尾的最长子序列的长度

每一层递归函数向上返回的时候,由于我们是以nums[i]为结尾的最长递增子序列,所以长度最小为1,就是nums[i]本身,所以返回的时候要+1,加的就是自身

然后在主函数中遍历一遍nums数组,把所有的i都传一遍,挑出里面的最大值

	int dfs(int i,vector<int>& nums)
    {
        int res=0;
        for(int j=0;j<i;j++)
            if(nums[j]<nums[i])
                res=max(res,dfs(j,nums));
        return res+1;
    }
    int lengthOfLIS(vector<int>& nums) {
        int res=0;
        for(int i=0;i<nums.size();i++)
            res=max(res,dfs(i,nums));
        return res;
    }

完整代码:

当然,这是超时的

class Solution {
public:
    int dfs(int i,vector<int>& nums)
    {
        int res=0;
        for(int j=0;j<i;j++)
            if(nums[j]<nums[i])
                res=max(res,dfs(j,nums));
        return res+1;
    }
    int lengthOfLIS(vector<int>& nums) {
        int res=0;
        for(int i=0;i<nums.size();i++)
            res=max(res,dfs(i,nums));
        return res;
    }
};

2.记忆化搜索

就是搞一个哈希表dp,全都初始化为-1,每次返回前给哈希表dp赋值,碰到不是-1的那就是算过的,那就直接返回计算过的结果,不需要再次递归了

class Solution {
public:
    int dfs(int i,vector<int>& nums,vector<int>& dp)
    {
        if(dp[i]!=0)
            return dp[i];
        int res=0;
        for(int j=0;j<i;j++)
            if(nums[j]<nums[i])
                res=max(res,dfs(j,nums,dp));
        return dp[i]=res+1;
    }
    int lengthOfLIS(vector<int>& nums) {
        vector<int> dp(nums.size(),0);
        int res=0;
        for(int i=0;i<nums.size();i++)
            res=max(res,dfs(i,nums,dp));
        return res;
    }
};

3.1:1翻译为动态规划

1.确定dp数组以及下标的含义

dp[i]就是以nums[i]结尾的最长递增子序列的长度

2.确定递推公式

和dfs中也是对应的

dp[i]=max(dp[i],dp[j]+1);

3.dp数组如何初始化

都初始化为1是因为nums[i]自身就算一个长度

vector<int> dp(nums.size(),1);

4.确定遍历顺序

后续结果需要依赖前面的计算结果,故使用从前往后遍历

然后在计算的时候记录一下最大值,最后返回最大值就好

		for(int i=0;i<nums.size();i++)
        {
            for(int j=0;j<i;j++)
                if(nums[j]<nums[i])
                    dp[i]=max(dp[i],dp[j]+1);
            if(dp[i]>res)
                res=dp[i];
        }

完整代码

class Solution {
public:
    int lengthOfLIS(vector<int>& nums) {
        vector<int> dp(nums.size(),1);
        int res=0;
        for(int i=0;i<nums.size();i++)
        {
            for(int j=0;j<i;j++)
                if(nums[j]<nums[i])
                    dp[i]=max(dp[i],dp[j]+1);
            if(dp[i]>res)
                res=dp[i];
        }
        return res;
    }
};

674.最长连续递增序列

674. 最长连续递增序列 - 力扣(LeetCode)

思路:

这个就比较简单了,选或不选的思路就行,dfs(i)表示以nums[i]结尾的最长连续递增子序列

我们nums[i-1]<nums[i]就选nums[i],否则的话就返回个1表示是nums[i]本身这个数是一个长度为1的子序列就好

dfs(i)=dfs(i-1)+1;

1.回溯法

class Solution {
public:
    int dfs(int i,vector<int>& nums)
    {
        if(i<0)
            return 0;
        if(i>0&&nums[i-1]<nums[i])
            return dfs(i-1,nums)+1;
        else
            return 1;
    }
    int findLengthOfLCIS(vector<int>& nums) {
        int res=0;
        for(int i=0;i<nums.size();i++)
            res=max(res,dfs(i,nums));
        return res;
    }
};

2.记忆化搜索

老样子还是返回前进行赋值就好

class Solution {
public:
    int dfs(int i,vector<int>& nums,vector<int>& dp)
    {
        if(dp[i]!=-1)
            return dp[i];
        if(i>0&&nums[i-1]<nums[i])
            return dp[i]=dfs(i-1,nums,dp)+1;
        else
            return dp[i]=1;
    }
    int findLengthOfLCIS(vector<int>& nums) {
        int res=0;
        vector<int> dp(nums.size(),-1);
        for(int i=0;i<nums.size();i++)
            res=max(res,dfs(i,nums,dp));
        return res;
    }
};

3.动态规划

class Solution {
public:
    int findLengthOfLCIS(vector<int>& nums) {
        int res=1;
        vector<int> dp(nums.size(),1);
        for(int i=1;i<nums.size();i++)
        {
            if(nums[i]>nums[i-1])
                dp[i]=dp[i-1]+1;
            res=max(res,dp[i]);
        }
        return res;
    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大数据学习15之Scala集合与泛型

1. 概述 大部分编程语言都提供了数据结构对应的编程库&#xff0c;并称之为集合库(Collection Library)&#xff0c;Scala 也不例外&#xff0c;且它还拥有以下优点&#xff1a; 易用&#xff1a;灵活组合运用集合库提供的方法&#xff0c;可以解决大部分集合问题 简洁&#xf…

Linux网络——网络初识

目录 1. 认识协议 2. 协议的分层 3. OSI 七层模型 && TCP/IP 五层(四层)模型 4. 网络传输的基本流程 5. 以太网的通信原理 6. 数据的跨网络传播 7. 认识 IP 地址 ① IP 是什么 ② IP 与 MAC 的关系 ③ 为什么需要 IP 在谈及网络之前&#xff0c;我们要先对学…

数字IC后端低功耗设计实现案例分享(3个power domain,2个voltage domain)

下图所示为咱们社区T12nm A55低功耗实现项目。其实这个项目还可以根据产品的需求做一些改进。改进后项目实现的难度会大大增加。也希望通过今天的这个项目案例分享&#xff0c;帮助到今年IC秋招的同学。 芯片低功耗设计实现upf编写指南&#xff08;附低功耗项目案例&#xff0…

Ubuntu从入门到精通(一)系统安装

Ubuntu从入门到精通&#xff08;一&#xff09; 1 Ubuntu镜像选择 下载Ubuntu 20.04系统ISO镜像 安装 Ubuntu 20.04系统,就必须有 Ubuntu 20.04系统软件安装程序可以通过浏览器访问Ubuntu20.04的官方站点&#xff0c; 然后在导舰栏找划 Dowwnloads->Mirrors链接&#xff…

说说软件工程中的“协程”

在软件工程中&#xff0c;协程&#xff08;coroutine&#xff09;是一种程序运行的方式&#xff0c;可以理解成“协作的线程”或“协作的函数”。以下是对协程的详细解释&#xff1a; 一、协程的基本概念 定义&#xff1a;协程是一组序列化的子过程&#xff0c;用户能像指挥家…

【linux】进程等待与进程替换

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;linux笔记仓 目录 01.进程等待系统调用获取子进程status常用宏使用示例 02.进程替换替换函数关键点解释&#xff1a;代码详细分析execvpe 函数的使用 01.进程等待 任何子进程&#xff0c;在退出的…

认证鉴权框架SpringSecurity-5--权限管理篇

上面两篇我们重点介绍了如何在代码上集成springSecurity&#xff0c;同时完成登录认证和token认证的过程。我们直到springSecurity处理能帮我们完成认证外&#xff0c;还可以帮助我们完成权限校验的工作&#xff0c;这篇我们来重点介绍下springSecurity是如何实现鉴权的。 一、…

RK3588开发板Android12-SDK更新通知

迅为RK3588开发板Android12 SDK升级至RK的android-12.1-mid-rkr14版本 内核版本&#xff1a;升级至 5.10.160 版本&#xff0c;提供更好兼容性和性能。 rkbin 版本&#xff1a;支持最新的 1.17 版本 bin 和 1.46 版本的 bl31。

stm32教程:OLED屏显示字母、汉字、图片工程讲解

早上好啊&#xff0c;大佬们&#xff0c;今天带来的是我们 stm32系列的第一个外设——OLED&#xff0c;相信大家对于OLED都不陌生了吧&#xff0c;这个可以说每一个项目里的必需品了&#xff0c;单片机离不开OLED就像西方离不开耶路撒冷。 在生活中&#xff0c;我们见到的OLED的…

力扣 LeetCode 28. 找出字符串中第一个匹配项的下标(Day4:字符串)

解题思路&#xff1a; KMP算法 需要先求得最长相等前后缀&#xff0c;并记录在next数组中&#xff0c;也就是前缀表&#xff0c;前缀表是用来回退的&#xff0c;它记录了模式串与主串(文本串)不匹配的时候&#xff0c;模式串应该从哪里开始重新匹配。 next[ j - 1 ] 记录了 …

我与Linux的爱恋:进程间通信 匿名管道

​ ​ &#x1f525;个人主页&#xff1a;guoguoqiang. &#x1f525;专栏&#xff1a;Linux的学习 文章目录 匿名管道pipe 匿名管道 匿名管道&#xff08;Anonymous Pipes&#xff09;是Unix和类Unix操作系统中的一种通信机制&#xff0c;用于在两个进程之间传递数据。匿名…

Java之JDBC,Maven,MYBatis

前言 就是用来操作数据库的 1.JDBC快速入门 注意在使用前一定要导入jar包 在模块那里新建目录&#xff0c;新建lib&#xff0c;粘贴复制jar包&#xff0c;我这个jar设置的是模块有效 package test1017;import java.sql.Connection; import java.sql.DriverManager; import…

基于Matlab的碎纸片的自动拼接复原技术

碎纸片的自动拼接复原技术 摘要&#xff1a;破碎文件的拼接在司法物证复原、历史文献修复以及军事情报获取等领域都有着重要的应用。目前发现对碎纸片的拼接大部分由人工完成&#xff0c;准确率较高&#xff0c;但耗费大量人力财力及时间&#xff0c;效率很低。随着计算机技术的…

STM32 设计的较为复杂的物联网项目,包括智能家居控制系统,涵盖了硬件和软件的详细设计。

使用 STM32 设计的较为复杂的物联网项目&#xff0c;包括智能家居控制系统&#xff0c;涵盖了硬件和软件的详细设计。 一、硬件设计 微控制器&#xff1a;选择 STM32F4 系列微控制器&#xff0c;如 STM32F407ZGT6&#xff0c;具有高性能和丰富的外设资源。 传感器模块&#x…

1.7 JS性能优化

从输入url到页面加载完成都做了些什么 输入 URL - 资源定位符 http://www.zhaowa.com - http 协议 域名解析 https://www.zhaowa.com > ip 1. 切HOST&#xff1f; > 浏览器缓存映射、系统、路由、运营商、根服务器 2. 实际的静态文件存放&#xff1f; 大流量 > 多个…

LPDDR4芯片学习(四)——DDR Training

一、ZQ Calibration DDR 学习时间 (Part B - 6)&#xff1a;DRAM ZQ 校正 - 知乎 (zhihu.com) 从原理上解释什么是DDR的ZQ校准&#xff1f; - 知乎 (zhihu.com) LPDDR4的训练(training)和校准(calibration)--ZQ校准(Calibration)_wonder_coole-腾讯云开发者社区 01 ZQ校准的…

pycharm分支提交操作

一、Pycharm拉取Git远程仓库代码 1、点击VCS > Get from Version Control 2、输入git的url&#xff0c;选择自己的项目路径 3、点击Clone&#xff0c;就拉取成功了 默认签出分支为main 选择develop签出即可进行开发工作 二、创建分支&#xff08;非必要可以不使用&#xf…

鸿蒙实战:页面跳转

文章目录 1. 实战概述2. 实现步骤2.1 创建项目2.2 准备图片素材2.3 编写首页代码2.4 创建第二个页面 3. 测试效果4. 实战总结 1. 实战概述 实战概述&#xff1a;本实战通过ArkUI框架&#xff0c;在鸿蒙系统上开发了一个简单的两页面应用。首页显示问候语和“下一页”按钮&…

IDEA部署AI代写插件

前言 Hello大家好&#xff0c;当下是AI盛行的时代&#xff0c;好多好多东西在AI大模型的趋势下都变得非常的简单。 比如之前想画一幅风景画得先去采风&#xff0c;然后写实什么的&#xff0c;现在你只需描述出你想要的效果AI就能够根据你的描述在几分钟之内画出一幅你想要的风景…

深入理解 Spark 中的 Shuffle

Spark 的介绍与搭建&#xff1a;从理论到实践_spark环境搭建-CSDN博客 Spark 的Standalone集群环境安装与测试-CSDN博客 PySpark 本地开发环境搭建与实践-CSDN博客 Spark 程序开发与提交&#xff1a;本地与集群模式全解析-CSDN博客 Spark on YARN&#xff1a;Spark集群模式…