动态规划课堂4-----子数组系列

目录

引入:

例题1:最大子数组和

例题2:环形子数组的最大和

例题3:乘积最大子数组

例题4:乘积为正数的最长子数组

总结:

结语:


引入:

在动态规划(DP)子数组系列中,我们还是用前面几节所用的解题思路1. 状态表示,2.状态转移方程,3.初始化,4.填表顺序,5.返回值。在写代码时一定要把这5步考虑清楚再写代码。写代码时其步骤也比较固定分别为:1.创建 dp 表 2.初始化 3.填表 4.返回值。写代码时可以按照这4步骤写不会乱也不会把哪一部分漏掉😎。

在子数组系列问题最常用到的状态表示是:以i位置元素为结尾的所有子数组的........(题目要求).

这个非常重要,后面的题基本都是用这个模板的状态表示。

例如下图:可以将其分为两种情况长度为1和长度大于1两种情况,剩下的分析要见具体题目。

例题1:最大子数组和

链接:最大子数组和

题目简介:

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

子数组:

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

解法(动态规划):

 1. 状态表示:

dp[i] 表示:以i 位置元素为结尾的「所有⼦数组」中和的最⼤和。设个状态表示会贯穿我们这一整个系列。

 2.状态转移方程

dp[i] 的所有可能可以分为以下两种:

(1)子数组的⻓度为1 :此时dp[i] = nums[i] ;(就只有它本身)

(2)子数组的⻓度⼤于1 :此时dp[i] 应该等于以i - 1 做结尾的「所有⼦数组」中和的最⼤值再加上nums[i] ,也就是dp[i - 1] + nums[i] 。

由于我们要的是「最⼤值」,因此应该是两种情况下的最⼤值,因此可得转移⽅程: dp[i] = max(nums[i], dp[i - 1] + nums[i]) 。

 3.初始化:

辅助结点法:

注意点:(1)辅助结点⾥⾯的值要保证后续填表是正确的.(2)下标的映射关系.

在本题中,最前⾯加上⼀个格⼦,并且让dp[0] = 0 即可。之所以设为0是因为根据1.状态表示,0位置没有元素(因为dp表加了一个辅助节点故dp表的0下标对应数组的-1下标),而我们的dp里面的值是最大和,没有数组故其和为0.

 4.填表顺序:

从左往右

 5.返回值:

状态表示为以i 为结尾的所有⼦数组的最⼤值,但是最⼤⼦数组和的结尾我们是不确定的(可能前面很大最后一个数非常小导致其不是最大值)。 因此我们需要返回整个dp 表中的最大值。

代码实现如下:

class Solution {public int maxSubArray(int[] nums) {//1.创建 dp 表//2.初始化//3.填表//4.返回值int n = nums.length;int[] dp = new int[n + 1];for(int i = 1;i <= n;i++){dp[i] = Math.max(nums[i - 1],dp[i - 1] + nums[i - 1]);}int max = dp[1];for(int i = 1;i <= n;i++){//加上辅助节点后要注意下标的映射关系,还有循环的边界。max = Math.max(max,dp[i]);}return max;}
}

时间复杂度:O(n)

空间复杂度:O(n)

例题2:环形子数组的最大和

链接:环形子数组的最大和

题目简介:

给定一个长度为 n 的环形整数数组 nums ,返回 nums 的非空 子数组 的最大可能和 

环形数组 意味着数组的末端将会与开头相连呈环状。形式上, nums[i] 的下一个元素是 nums[(i + 1) % n] , nums[i] 的前一个元素是 nums[(i - 1 + n) % n] 。

子数组 最多只能包含固定缓冲区 nums 中的每个元素一次。形式上,对于子数组 nums[i], nums[i + 1], ..., nums[j] ,不存在 i <= k1, k2 <= j 其中 k1 % n == k2 % n 。

 解法(动态规划):

我们可以将最大和子数组大致划分为下面两个部分,一个是正常在中间的,另一个在头尾两边的。

这一题是例题1的升级版,如果有友友分析过的话会发现直接进行规划非常困难,那么我们能不能采用另一种思维。既然整个数组的和sum是不变的那么要求头尾两边的子数组我们能不能求正常中间数组的最小值将sum减去这个最小值不就是我们头尾两边的子数组的最大值。

 1. 状态表示:

f[i] 表⽰:以i 做结尾的「所有⼦数组」中和的最大值。

g[i] 表⽰:以i 做结尾的「所有⼦数组」中和的最小值。

 2.状态转移方程: 

由于f[i]在上一题已经分析过故这里只分析g[i].

(1)子数组的⻓度为1 :此时g[i] = nums[i] ;

(2)子数组的⻓度⼤于1 :此时g[i] 应该等于以i - 1 做结尾的所有⼦数组中和的最⼩值再加上nums[i] ,也就是g[i - 1] + nums[i] 。

由于我们要的是最小子数组和,因此应该是两种情况下的最⼩值,因此可得转移⽅程:g[i] = min(nums[i], g[i - 1] + nums[i]) 。

 3.初始化:

辅助节点法.

前面加上⼀个格子,并且让g[0] = 0,f[0] = 0,根据状态表示为0,没有元素那么和显然为0.

 4.填表顺序:

从左往右

 5.返回值:

按正常情况来我们应该返回f()里面的最大值,和g表里面的最小值-sum,但是有一种特殊的情况如下图,就是如果数组的数全为负数的情况下,g表会把所有的负数都加入,然后sum-g表就会等于0,下面的正确最大和为-1sum-g表最小值等于0的情况就相当于一个子数组里面一个数都没有这是题目不允许的

故我们返回sum == gmin ? fmax : max(fmax, sum - gmin)。这样就可以排除全为负数的情况。

代码实现如下:

class Solution {public int maxSubarraySumCircular(int[] nums) {//1.创建 dp 表//2.初始化//3.填表//4.返回值int n = nums.length;int sum = 0;for(int i = 0;i < n;i++){sum += nums[i];}int[] f = new int[n + 1];int[] g = new int[n + 1];for(int i = 1;i <= n;i++){f[i] = Math.max(nums[i - 1],nums[i - 1] + f[i - 1]);g[i] = Math.min(nums[i - 1],nums[i - 1] + g[i - 1]);}int max1 = f[1];for(int i = 1;i <= n;i++){max1 = Math.max(max1,f[i]);}int max2 = -0x3f3f3f3f;for(int i = 1;i <= n;i++){if(sum == g[i]){max2 = Math.max(max2,-0x3f3f3f3f);}else{max2 = Math.max(sum - g[i],max2);}}return Math.max(max1,max2);}
}

max2初始化为-0x3f3f3f3f是因为求最大值初始化的值要够小,之所以不初始化为Integer.MIN_VALUE,这是因为这个数如果再发生减法的话可能会变成无穷大,0x3f3f3f3f这个数够小且稳定。

时间复杂度:O(n)

空间复杂度:O(n)

例题3:乘积最大子数组

链接:乘积最大子数组

题目简介:

给你一个整数数组 nums ,请你找出数组中乘积最大的非空连续子数组(该子数组中至少包含一个数字),并返回该子数组所对应的乘积。

测试用例的答案是一个 32-位 整数。

解法(动态规划):

这题乍一看好像和最大子数组和好像差不多,但是乘法要考虑正负号的因素,如果只有一个dp表存储表示乘积最大子数组那么下面这种情况的最大值只能是5,可是结果是30.所以只有一个dp表是不够的,我们还需要一个表来存储最小值(乘于一个负数就变成最大值)。

 1. 状态表示:

f[i] 表⽰:以i 结尾的所有⼦数组的最大乘积。

g[i] 表⽰:以i 结尾的所有⼦数组的最小乘积。

 2.状态转移方程:

对于f[i] ,也就是「以i 为结尾的所有⼦数组的最⼤乘积」,对于所有⼦数组,可以分为下面三种形式:

(1)子数组的⻓度为1 ,也就是nums[i]。

(2)子数组的⻓度⼤于1 ,但nums[i] > 0 ,此时需要的是i - 1 为结尾的所有⼦数组的最⼤乘积f[i - 1] ,再乘上nums[i] ,也就是nums[i] * f[i - 1] 。

(3)子数组的⻓度⼤于1 ,但nums[i] < 0 ,此时需要的是i - 1 为结尾的所有⼦数组的最⼩乘积g[i - 1] ,再乘上nums[i] ,也就是nums[i] * g[i - 1] 。

如果nums[i] = 0 ,所有⼦数组的乘积均为0 ,三种情况其实都包含了 。

综上所述, f[i] = max(nums[i], max(nums[i] * f[i - 1], nums[i] * g[i - 1]) )。

对于g[i] ,也就是「以i 为结尾的所有子数组的最⼩乘积」,对于所有⼦数组,可以分为下面三种形式:

(1)子数组的⻓度为1 ,也就是nums[i] 。

(2)子数组的⻓度⼤于1 ,但nums[i] > 0 ,此时需要的是i - 1 为结尾的所有⼦数组的最⼩乘积g[i - 1] ,再乘上nums[i] ,也就是nums[i] * g[i - 1] 。

(3)子数组的⻓度⼤于1 ,但nums[i] < 0 ,此时需要的是i - 1 为结尾的所有⼦数组的最⼤乘积f[i - 1] ,再乘上nums[i] ,也就是nums[i] * f[i - 1] 。

综上所述, g[i] = min(nums[i], min(nums[i] * f[i - 1], nums[i] * g[i - 1])) 。

 3.初始化:

辅助节点:

f[0] = g[0] = 1。

 4.填表顺序:

从左往右,两个表⼀起填。

 5.返回值:

返回f 表中的最⼤值

代码如下:

class Solution {public int maxProduct(int[] nums) {//1.创建 dp 表//2.初始化//3.填表//4.返回值int n = nums.length;int[] f = new int[n + 1];int[] g = new int[n + 1];g[0] = f[0] = 1;for(int i = 1;i <= n;i++){f[i] = nums[i - 1];g[i] = nums[i - 1];if(nums[i - 1] >= 0){f[i] = Math.max(f[i],nums[i - 1] * f[i - 1]);g[i] = Math.min(nums[i - 1] * g[i - 1],g[i]);}else{f[i] = Math.max(f[i],nums[i - 1] * g[i - 1]);g[i] = Math.min(g[i],nums[i - 1] * f[i - 1]);}}int max = f[1];for(int i = 1;i <= n;i++){max = Math.max(max,f[i]);}return max;}
}

时间复杂度:O(n)

空间复杂度:O(n)

例题4:乘积为正数的最长子数组

链接:乘积为正数的最长子数组

题目简介:

解法(动态规划):

但是,如果我们知道「以i - 1 为结尾的所有子数组,乘积为负数的最长子数组的⻓度」neg[i - 1] ,那么此时的dp[i] 是不是就等于neg[i - 1] + 1呢?

通过上⾯的分析,我们可以得出,需要两个dp 表,才能推导出最终的结果。不仅需要⼀个乘积为正数的最长子数组,还需要⼀个乘积为负数的最长子数组。

 1. 状态表示:

f[i] 表⽰:以i 结尾的所有⼦数组中,乘积为正数的最长子数组的⻓度。

g[i] 表⽰:以i 结尾的所有⼦数组中,乘积为负数的最长子数组的⻓度。

 2.状态转移方程:

推状态转移方程时最好画图来帮助我们理解。

对于f[i] ,也就是以i 为结尾的乘积为正数的最⻓⼦数组,根据nums[i] 的值,可以分为三种情况:

(1)nums[i] = 0 时,所有以i 为结尾的⼦数组的乘积都不可能是正数,此时f[i] = 0 .

(2)nums[i] > 0 时,那么直接找到f[i - 1] 的值(这⾥请再读⼀遍f[i - 1] 代表的意义,并且考虑如果f[i - 1] 的结值是0的话,影不影响结果),然后加⼀即可, 此时f[i] = f[i - 1] + 1.

(3)nums[i] < 0 时,此时我们要看g[i - 1] 的值(这⾥请再读⼀遍g[i - 1] 代表的意义。因为负负得正,如果我们知道以i - 1 为结尾的乘积为负数的最⻓⼦数组的⻓度,加上1即可),根据g[i - 1] 的值,⼜要分两种情况:

1.g[i - 1] = 0 ,说明以i - 1 为结尾的乘积为负数的最长子数组是不存在的,⼜因为nums[i] < 0,所以以i 结尾的乘积为正数的最长子数组也是不存在的,此时f[i] = 0。

2.g[i - 1] != 0 ,说明以i - 1 为结尾的乘积为负数的最长子数组是存在的,⼜因为nums[i] < 0 ,所以以i 结尾的乘积为正数的最长子数组就等于g[i - 1] + 1。

综上所述, nums[i] < 0 时, f[i] = g[i - 1] == 0 ? 0 : g[i - 1] + 1;

 3.初始化:

辅助节点法:

f[0] = g[0] = 0。

 4.填表顺序:

从左往右,两个表⼀起填。

 5.返回值: 

根据状态表⽰,我们要返回f 表中的最⼤值。

代码如下:

class Solution {public int getMaxLen(int[] nums) {//1.创建 dp 表//2.初始化//3.填表//4.返回值int n = nums.length;int[] f = new int[n + 1];int[] g = new int[n + 1];for(int i = 1;i <= n;i++){if(nums[i - 1] > 0){f[i] = f[i - 1] + 1;g[i] = (g[i - 1] == 0) ? 0 : g[i - 1] + 1;}else if(nums[i - 1] < 0){f[i] = (g[i - 1] == 0) ? 0 : g[i - 1] + 1;g[i] = f[i - 1] + 1;}else{f[i] = 0;g[i] = 0;}}int max = f[1];for(int i = 1;i <= n;i++){max = Math.max(max,f[i]);}return max;}
}

时间复杂度:O(n)

空间复杂度:O(n)

总结:

由于文章篇幅有限我就挑选了经典的关于子数组dp问题来讲解,还有一些题我觉得也很好但写不下了😭,希望大家做完四题后还要多练练,下面我再给出两题有兴趣的友友可以AK。

题目一:最长湍流子数组

题目二:单词拆分

结语:

其实写博客不仅仅是为了教大家,同时这也有利于我巩固知识点,和做一个学习的总结,由于作者水平有限,对文章有任何问题还请指出,非常感谢。如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

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

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

相关文章

java中移位<< >> <<< |数据类型转换

移位 x64转换二进制&#xff1a;100 0000 左移2位 &#xff1a; 1000 0000 0 对应十进制 i 256 >>右移 <<左移 >>无符号位右移 关于右移一位相当于整除2 数据类型及其转换 基本数据类型&#xff0c;数据类型范围 byte(-128~127)&#xff08;-2^7~2…

【开源】SpringBoot框架开发教学资源共享平台

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 课程档案模块2.3 课程资源模块2.4 课程作业模块2.5 课程评价模块 三、系统设计3.1 用例设计3.2 类图设计3.3 数据库设计3.3.1 课程档案表3.3.2 课程资源表3.3.3 课程作业表3.3.4 课程评价表 四、系统展…

mysql如何开启手动提交事务

在mysql中&#xff0c;有一个变量autocommit&#xff0c;表示自动提交&#xff0c;默认为1&#xff0c;表示开启自动提交。通过以下命令查询 select autocommit;当autocommit为1时&#xff0c;任何一条sql语句都是一个事务&#xff0c;执行完由mysql自动提交。如果想自己决定什…

Java 抽象类和接口

登神长阶 第三阶 抽象类和接口 &#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&#x1f340;&…

LVS负载均衡集群基础概念

目录 一、集群 1、集群概述 1.1 什么是集群 1.2 集群系统扩展方式 1.2.1 Scale UP&#xff08;纵向扩展&#xff09; 1.2.2 Scale OUT&#xff08;横向扩展&#xff09; 1.2.3 区别 1.3 分布式系统 1.4 分布式与集群 1.5 集群设计原则 1.6 集群设计实现 1.6.1 基础…

存储引擎的简介

简介&#xff1a; 1.在mysql存储引擎可以说就是指表的类型&#xff0c;可以称为表处理器&#xff0c;以表的形式存储。 2.他的功能就是接收上层传下来的指令&#xff0c;然后对表中的数据进行提取写入操作。 目的&#xff1a; 为了管理方便&#xff0c;我们把连接管理&#xf…

并查集(蓝桥杯 C++ 题目 代码 注解)

目录 介绍&#xff1a; 模板&#xff1a; 题目一&#xff08;合根植物&#xff09;&#xff1a; 代码&#xff1a; 题目二&#xff08;蓝桥幼儿园&#xff09;&#xff1a; 代码&#xff1a; 题目三&#xff08;小猪存钱罐&#xff09;&#xff1a; 代码&#xff1a; …

AWS的CISO:GenAI只是一个工具,不是万能钥匙

根据CrowdStrike的年度全球威胁报告,尽管研究人员预计人工智能将放大对防御者和攻击者的影响,但威胁参与者在其行动中使用人工智能的程度有限。该公司上个月在报告中表示:“在整个2023年,很少观察到GenAI支持恶意计算机网络运营的开发和/或执行。” 对于GenAI在网络安全中的…

《vtk9 book》 官方web版 第3章 - 计算机图形基础 (3 / 6)

3.8 演员几何 我们已经看到了光照属性如何控制演员的外观&#xff0c;以及相机如何结合变换矩阵将演员投影到图像平面上。剩下的是定义演员的几何形状&#xff0c;以及如何将其定位在世界坐标系中。 建模 计算机图形学研究中的一个重要主题是建模或表示物体的几何形状。…

贪心算法(蓝桥杯 C++ 题目 代表 注解)

介绍&#xff1a; 贪心算法&#xff08;Greedy Algorithm&#xff09;是一种在每一步选择中都采取当前状态下最好或最优&#xff08;即最有利&#xff09;的选择&#xff0c;从而希望最终能够得到全局最好或最优的结果的算法。它通常用来解决一些最优化问题&#xff0c;如最小生…

扩展CArray类,增加Contain函数

CArray不包含查找类的函数&#xff0c;使用不便。考虑扩展CArray类&#xff0c;增加Contain函数&#xff0c;通过回调函数暴露数组元素的比较方法&#xff0c;由外部定义。该方法相对重载数组元素的“”符号更加灵活&#xff0c;可以根据需要配置不同的回调函数进行比较 //类型…

电脑解锁后黑屏有鼠标--亲测!!不需要重装系统!!

问题&#xff1a;上周电脑黑屏&#xff0c;只有鼠标&#xff0c;鼠标还不能右键&#xff01;&#xff01; 中招&#xff1a;win10系统最新版火绒安全 &#xff0c;那你有概率获得开机黑屏套餐一份。 原因是&#xff1a;火绒把我们的explorer删除了导致黑屏&#xff0c;这个文…

Synthetic Temporal Anomaly Guided End-to-End Video Anomaly Detection 论文阅读

Synthetic Temporal Anomaly Guided End-to-End Video Anomaly Detection 论文阅读 Abstract1. Introduction2. Related Work3. Methodology3.1. Architecture3.1.1 Autoencoder3.1.2 Temporal Pseudo Anomaly Synthesizer 3.2. Training3.3. Anomaly Score 4. Experiments4.1.…

每日学习笔记:C++ STL 的Array

Array定义 Array模板有两个参数&#xff0c;一个是元素类型&#xff0c;一个是数组大小 Array初始化 Array的操作 Array当作C数组 Array的Tuple接口

接口自动化测试从入门到高级实战!

接口测试背景和必要性 接口测试是测试系统组件间接口&#xff08;API&#xff09;的一种测试&#xff0c;主要用于检测内部与外部系统、内部子系统之间的交互质量&#xff0c;其测试重点是检查数据交换、传递的准确性&#xff0c;控制和交互管理过程&#xff0c;以及系统间相互…

【Selenium】selenium介绍及工作原理

一、Selenium介绍 用于Web应用程序测试的工具&#xff0c;Selenium是开源并且免费的&#xff0c;覆盖IE、Chrome、FireFox、Safari等主流浏览器&#xff0c;通过在不同浏览器中运行自动化测试。支持Java、Python、Net、Perl等编程语言进行自动化测试脚本编写。 官网地址&…

伪分布Hadoop的安装与部署

1.实训目标 &#xff08;1&#xff09;熟悉掌握使用在Linux下安装JDK。 &#xff08;2&#xff09;熟悉掌握使用在Linux下安装Hadoop。 &#xff08;3&#xff09;熟悉掌握使用配置SSH免密登录。 2.实训环境与软件 环境 版本 说明 Windows 10系统 64位 操作电脑配置 …

ChatGPT 串接到 Discord - 团队协作好助理

ChatGPT 串接到 Discord - 团队协作好助理 ChatGPT 是由 OpenAI 开发的一个强大的语言模型&#xff0c;本篇文章教你如何串接 Discord Bot &#xff0c;协助团队在工作上更加高效并促进沟通与协作。使 ChatGPT 发挥出最大的功效&#xff0c;进一步提升工作效率和团队协作能力。…

二 超级数据查看器   讲解稿   导入功能

二 超级数据查看器 讲解稿 导入功能 APP下载地址 百度手机助手 下载地址4 ​ 讲解稿全文&#xff1a; 大家好。 今天我们对 超级数据查看器的 导入信息功能 做一下详细讲解。 首先&#xff0c;我们打开 超级数据查看器。 我们这个系统要实现的是&#xff0c;快速生…

[Electron]中IPC进程间通信

Electron中IPC 进程间通信 (IPC) 是在 Electron 中构建功能丰富的桌面应用程序的关键部分之一。在 Electron 中&#xff0c;进程使用 ipcMain 和 ipcRenderer 模块&#xff0c;通过开发人员定义的“通道”传递消息来进行通信。 本文介绍以下几个方面&#xff1a; 1-渲染进程到…