【LeetCode: 1911. 最大子序列交替和 | 暴力递归=>记忆化搜索=>动态规划 】

news2024/12/26 21:50:09

在这里插入图片描述

🚀 算法题 🚀

🌲 算法刷题专栏 | 面试必备算法 | 面试高频算法 🍀
🌲 越难的东西,越要努力坚持,因为它具有很高的价值,算法就是这样✨
🌲 作者简介:硕风和炜,CSDN-Java领域优质创作者🏆,保研|国家奖学金|高中学习JAVA|大学完善JAVA开发技术栈|面试刷题|面经八股文|经验分享|好用的网站工具分享💎💎💎
🌲 恭喜你发现一枚宝藏博主,赶快收入囊中吧🌻
🌲 人生如棋,我愿为卒,行动虽慢,可谁曾见我后退一步?🎯🎯

🚀 算法题 🚀

在这里插入图片描述
在这里插入图片描述

🍔 目录

    • 🚩 题目链接
    • ⛲ 题目描述
    • 🌟 求解思路&实现代码&运行结果
      • ⚡ 暴力法
        • 🥦 求解思路
        • 🥦 实现代码
        • 🥦 运行结果
      • ⚡ 记忆化搜索
        • 🥦 求解思路
        • 🥦 实现代码
        • 🥦 运行结果
      • ⚡ 动态规划
        • 🥦 求解思路
        • 🥦 实现代码
        • 🥦 运行结果
    • 💬 共勉

🚩 题目链接

  • 1911. 最大子序列交替和

⛲ 题目描述

一个下标从 0 开始的数组的 交替和 定义为 偶数 下标处元素之 和 减去 奇数 下标处元素之 和 。

比方说,数组 [4,2,5,3] 的交替和为 (4 + 5) - (2 + 3) = 4 。
给你一个数组 nums ,请你返回 nums 中任意子序列的 最大交替和 (子序列的下标 重新 从 0 开始编号)。

一个数组的 子序列 是从原数组中删除一些元素后(也可能一个也不删除)剩余元素不改变顺序组成的数组。比方说,[2,7,4] 是 [4,2,3,7,2,1,4] 的一个子序列(加粗元素),但是 [2,4,2] 不是。

示例 1:

输入:nums = [4,2,5,3]
输出:7
解释:最优子序列为 [4,2,5] ,交替和为 (4 + 5) - 2 = 7 。
示例 2:

输入:nums = [5,6,7,8]
输出:8
解释:最优子序列为 [8] ,交替和为 8 。
示例 3:

输入:nums = [6,2,1,2,4,5]
输出:10
解释:最优子序列为 [6,1,5] ,交替和为 (6 + 5) - 1 = 10 。

提示:

1 <= nums.length <= 105
1 <= nums[i] <= 105

🌟 求解思路&实现代码&运行结果


⚡ 暴力法

🥦 求解思路

  1. 简单概括题目的意思:这是一道子序列类型的题目,每个位置可以选择,也可以不选,让我们求得nums数组中任意子序列的最大交替和。(最大交替和的定义请看题目)
  2. 具体怎么求解呢?我们对题目进行一个拆分,发现最终求解的问题规模是可以转换为更小的问题规模进行求解的,所以我们通过递归的思路来求解。
  3. 问题又来了?怎么设计递归函数呢?这个就需要大家认真考虑了。
  4. 我们假设定义递归函数process(index,flag),函数的含义是从index位置开始,flag=0代表此时已经选择了偶数个数,flag=1则相反,最终返回最大交替和。
  5. 具体的状态是怎么转移的呢?我们可以分为以下俩种情况:如果当前选择了偶数个数,那么此时有俩种可能,可以之前并没有选择;也有可能是之前选择了奇数个数并且选择当前的数,俩者去较大值即可;同理可以推出如果选择了奇数个数的情况,最终再取得俩种情况的最大值即可。
  6. 有了基本的思路,接下来我们就来通过代码来实现一下。

🥦 实现代码

class Solution {

    int[] nums;
    int n;

    public long maxAlternatingSum(int[] nums) {
        n=nums.length;
        this.nums=nums;
        return process(0,0);
    }

    public long process(int index,int cnt){
        if(index>=n) return 0;
        long p1=0,p2=0;
        if(cnt==0){
            p1=Math.max(process(index+1,0),process(index+1,1)+nums[index]);
        }else{
            p2=Math.max(process(index+1,1),process(index+1,0)-nums[index]);
        }
        return Math.max(p1,p2);
    }
}

🥦 运行结果

时间复杂度&空间复杂度

在这里插入图片描述


⚡ 记忆化搜索

🥦 求解思路

  1. 因为在递归的过程中,会重复的出现一些多次计算的结果,我们通过开辟一个数组,将结果提前缓存下来,算过的直接返回,避免重复计算,通过空间来去换我们的时间。

🥦 实现代码

class Solution {

    int[] nums;
    int n;
    long[][] map;

    public long maxAlternatingSum(int[] nums) {
        n=nums.length;
        this.nums=nums;
        map=new long[n+1][2];
        for(int i=0;i<=n;i++){
            for(int j=0;j<2;j++){
                map[i][j]=-1;
            }
        }
        return process(0,0);
    }

    public long process(int index,int cnt){
        if(index>=n) return 0;
        if(map[index][cnt]!=-1) return map[index][cnt];
        long p1=0,p2=0;
        if(cnt==0){
            p1=Math.max(process(index+1,0),process(index+1,1)+nums[index]);
        }else{
            p2=Math.max(process(index+1,1),process(index+1,0)-nums[index]);
        }
        map[index][cnt]=Math.max(p1,p2);
        return map[index][cnt];
    }
}

🥦 运行结果

通过缓存,将重复计算的结果缓存下来,通过。

时间复杂度&空间复杂度
在这里插入图片描述


⚡ 动态规划

🥦 求解思路

  1. 有了递归,有了记忆化搜索,接下来就是动态规划了,直接上手。

🥦 实现代码

class Solution {

    int[] nums;
    int n;
    long[][] dp;

    public long maxAlternatingSum(int[] nums) {
        this.n=nums.length;
        this.nums=nums;
        dp=new long[n+1][2];
        dp[n][0]=dp[n][1]=0;
        for(int index=n-1;index>=0;index--){
            dp[index][0]=Math.max(dp[index+1][0],dp[index+1][1]+nums[index]);
            dp[index][1]=Math.max(dp[index+1][1],dp[index+1][0]-nums[index]);
        }
        return dp[0][0];
    }
}

🥦 运行结果

动态规划搞定,大家可以积极的尝试。

时间复杂度&空间复杂度

在这里插入图片描述

扩展:还可以继续进行优化算法的空间复杂度


💬 共勉

最后,我想和大家分享一句一直激励我的座右铭,希望可以与大家共勉!

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

【C#】设置输入法,解决扫描枪在中文状态下识别异常问题

系列文章 【C#】编号生成器&#xff08;定义单号规则、固定字符、流水号、业务单号&#xff09; 本文链接&#xff1a;https://blog.csdn.net/youcheng_ge/article/details/129129787 【C#】日期范围生成器&#xff08;开始日期、结束日期&#xff09; 本文链接&#xff1a;h…

【Redis】特殊数据类型 Stream (流)

&#x1f3af;前言 除了五中基本的数据类型外&#xff0c;Redis还支持两种特殊的数据类型&#xff0c;第一种 Geo (地理位置)&#xff1a;用于存储地理位置相关的数据&#xff0c;例如经纬度、距离等。第二种 Stream (流)&#xff1a;是一个高级的列表类型&#xff0c;支持对列…

学习系统编程No.30【多线程控制实战】

引言&#xff1a; 北京时间&#xff1a;2023/7/7/9:58&#xff0c;耳机正在充电中&#xff0c;所以刚好让我们先把引言写一写&#xff0c;昨天睡觉前听了一会小说&#xff0c;听小说的好处就在于&#xff0c;它可以让你放下手机&#xff0c;快速睡觉&#xff0c;并且还有一定的…

python批量检测网站是否能打开

import requestsif __name__ "__main__":file_name input() #读取文件名fp1 open(file_name, "r") #以只读&#xff0c;打开文件for line in fp1.readlines(): #readlines 按行读取文件&#xff0c;会保留\n&#xff0c;返回一个(文…

3. MySQL - 数据类型 选项约束

目录 回顾 1. 命令行下的 MySql 客户端 2. 图形化界面的 MySQL-Client 3. 数据库概述 3.1 数据库管理系统是什么 3.2 工作模式 3.3 RDBMS 管理数据的结构 3.4 客户端连接服务器的信息 4. MySQL 中的数据类型 4.1 整型类型 4.2 字符串 4.3 日期/时间 5. MySQL 每个字…

【Ubuntu18.04 解决蓝牙wifi 之ax201无线网卡驱动安装】

【Ubuntu18.04 解决蓝牙wifi 之ax201无线网卡驱动安装】 1. 前言2. 更新linux内核 3. 下载安装intel ax201网卡驱动 1. 前言 台式机安装了双系统win11Ubuntu18.04系统&#xff0c;发现没有无线网卡和蓝牙&#xff0c;经查阅资料发现由于网卡刚没多久&#xff0c;Ubuntu没有集成…

服务器反向代理

反向代理作用 隐藏服务器信息 -> 保证内网的安全&#xff0c;通常将反向代理作为公网访问地址&#xff0c;web服务器是内网&#xff0c;即通过nginx配置外网访问web服务器内网 举例 百度的网址是&#xff1a;http://www.baidu.com &#xff0c; 现在我通过自己的服务器地…

Figma源文件导出技巧:提升效率的简易步骤

因为Figma&#xff0c;sketch,xd都支持导入sketch格式,所以我们只要将文件格式转成sketch&#xff0c;就能自由的在不同软件间导入导出。 现在就有一个网站可以帮助你快速简单的导入Figma、Sketch、XD 等格式文件&#xff0c;&#xff0c;还可以导出 Sketch 文件满足跨工具协作…

Nacos 服务注册和配置中心

文章目录 1 应用1.1 依赖1.2 配置文件 2 Nacos发现实例模型3 注册中心对比4 Nacos 支持AP和CP模式的切换4.1 何时选择何种模式&#xff1f; 5 Nacos 服务配置10.5.1 SpringCloud原生注解RefreshScope5.2 配置5.3 分类设计思想 6 Nacos 集群是持久化配置6.1 Nacos支持三种部署模…

Redis实战案例16-redisson的快速入门

1. 可能存在的问题 不可重入&#xff1a;基于SETNX实现的简单分布式锁通常不支持可重入性&#xff0c;即同一个客户端在获取锁后不能再次获取锁&#xff0c;否则会导致死锁。不可重试&#xff1a;如果多个客户端同时尝试获取锁但都失败了&#xff0c;并且没有重试机制&#xff…

基于RWKV-Runner大语言模型系统

RWKV Runner 旨在消除大语言模型的使用门槛,全自动处理AI对话,并且提供了OpenAI API兼容的接口。使用起来简单方便,但是还是比较吃机器,显存2G到32G都可以使用,根据自己的模型选择即可。 总结起来: 使用起来方便简单,上手容易。需要有电脑基础,很多地方还不是傻瓜化。需…

LayUI框架——选项卡等element组件使用

目录 前言 一、element组件 1. element基础方法 2. 更新渲染 二、动态实现选项卡 要求 1. 优化dao类 2. 优化前端JSP页面 3. 引入头部hand.jsp页面 4. 优化后台主界面js 5. 运行效果图 前言 在项目中我们需要编写许多页面&#xff0c;在页面中有许多元素需要自动去完…

【Unity编辑器扩展】字库裁剪工具, 优化字体文件大小,批量修改文本组件字体

原理&#xff1a; 1. 扫描项目中用到的字符集&#xff1b; 2. 把字体文件裁剪掉没用到的字符&#xff0c;仅保留项目中用到的字符&#xff1b; 3. 生成裁剪后的字体文件&#xff1b; 工具功能设计&#xff1a; 1. 支持通过拖拽字体文件或文件夹批量选择需要裁剪的字体文件。…

网络安全设备Bypass功能介绍及分析

网络安全平台厂商往往需要用到一项比较特殊的技术&#xff0c;那就是Bypass&#xff0c;那么到底什么是Bypass呢&#xff0c;Bypass设备又是如何来实现的&#xff1f;下面我就对Bypass技术做一下简单的介绍和说明。 一、 什么是Bypass。 大家知道&#xff0c;网络安全设备一般…

mac安装Golang开发环境及入门

目录 一、Mac brew 安装go环境 1.1 安装步骤 1.2 设置GOPATH 及环境变量 1.3 编写第一个go程序 二、快速入门 1.1 快速入门需求 1.2 go学习&#xff08;自用&#xff09; 一、Mac brew 安装go环境 1.1 安装步骤 1&#xff09;终端输入&#xff0c;也可以指定下载go版本…

SPSSAU方差分析+python

准备数据 将数据格式调整为以下格式&#xff1a; jupyter处理过程 #读取数据 import numpy as np import pandas as pd# 创建一个空的DataFrame t1 pd.DataFrame() t2 pd.DataFrame() t3 pd.DataFrame() T1pd.read_excel(./数据/抑郁_T1.xlsx)T1.columnsT1.iloc[0] T1T1…

模板类与继承

模板类与继承 模板类继承普通类普通类继承模板类的实例化版本。普通类继承模板类模板类继承模板类模板类继承模板参数给出的基类 模板类继承普通类 基类 派生类 测试函数; 普通类继承模板类的实例化版本。 模板基类 普通类继承模板基类的实例化版本&#xff1a; 普通…

PROFINET转DeviceNet网关devicenet通讯模块

远创智控YC-DNT-PN这款神器&#xff0c;连接PROFINET和DeviceNet网络&#xff0c;让两边数据轻松传输。 这个网关不仅从ETHERNET/IP和DEVICENET一侧读写数据&#xff0c;还可以将缓冲区数据交换&#xff0c;这样就可以在两个网络之间愉快地传递数据了&#xff01;而且&#xff…

云计算的学习(三)

三、云计算中的网络基础知识 3.1虚拟化中网络的架构 a.虚拟化中网络的架构 二层交换机作为接入交换机使用&#xff0c;三层交换机可以作为汇聚交换机或核心交换机&#xff0c;在抛开网络安全设备时&#xff0c;路由器直接连接在互联网上。 b.广播和单播 物理服务器内部主要…