最大子数组和[中等]

一、题目

给定一个长度为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:
输入:nums = [1,-2,3,-2]
输出:3
解释:从子数组[3]得到最大和3

示例 2:
输入:nums = [5,-3,5]
输出:10
解释:从子数组[5,5]得到最大和5 + 5 = 10

示例 3:
输入:nums = [3,-2,2,-3]
输出:3
解释:从子数组[3][3,-2,2]都可以得到最大和3

n == nums.length
1 <= n <= 3 * 104
-3 * 104 <= nums[i] <= 3 * 104​​​​​​​

二、代码

【1】动态规划: 求解普通数组的最大子数组和是求解环形数组的最大子数组和问题的子集。设数组长度为n,下标从0开始,在环形情况中,答案可能包括以下两种情况:
1、构成最大子数组和的子数组为nums[i:j],包括nums[i]nums[j−1]j−i个元素,其中0≤i<j≤n
2、构成最大子数组和的子数组为nums[0:i]nums[j:n],其中0<i<j<n

第二种情况中,答案可以分为两部分,nums[0:i]为数组的某一前缀,nums[j:n]为数组的某一后缀。求解时,我们可以枚举j,固定sum(nums[j:n])的值,然后找到右端点坐标范围在[0,j−1]的最大前缀和,将它们相加更新答案。

右端点坐标范围在[0,i]的最大前缀和可以用leftMax[i]表示,递推方程为:leftMax[i]=max⁡(leftMax[i−1],sum(nums[0:i+1])

至此,我们可以使用以上方法求解出环形数组的最大子数组和。特别需要注意的是,本题要求子数组不能为空,我们需要在代码中做出相应的调整。

class Solution {public int maxSubarraySumCircular(int[] nums) {int n = nums.length;int[] leftMax = new int[n];// 对坐标为 0 处的元素单独处理,避免考虑子数组为空的情况leftMax[0] = nums[0];int leftSum = nums[0];int pre = nums[0];int res = nums[0];for (int i = 1; i < n; i++) {pre = Math.max(pre + nums[i], nums[i]);res = Math.max(res, pre);leftSum += nums[i];leftMax[i] = Math.max(leftMax[i - 1], leftSum);}// 从右到左枚举后缀,固定后缀,选择最大前缀int rightSum = 0;for (int i = n - 1; i > 0; i--) {rightSum += nums[i];res = Math.max(res, rightSum + leftMax[i - 1]);}return res;}
}

时间复杂度: O(n),其中nnums的长度。求解第一种情况的时间复杂度为O(n),求解leftMax数组和枚举后缀的时间复杂度为O(n),因此总的时间复杂度为O(n)
空间复杂度: O(n),其中nnums的长度。过程中我们使用leftMax来存放最大前缀和。

【2】取反: 对于第二种情况,即环形数组的最大子数组和为nums[0:i]nums[j:n],我们可以找到普通数组最小的子数组nums[i:j]即可。而求解普通数组最小子数组和的方法与求解最大子数组和的方法完全相同。

maxRes是普通数组的最大子数组和,minRes是普通数组的最小子数组和,我们可以将maxRes∑i=0nnums[i]−minRes取最大作为答案。

需要注意的是,如果maxRes<0,数组中不包含大于等于0的元素,minRes将包括数组中的所有元素,导致我们实际取到的子数组为空。在这种情况下,我们只能取maxRes作为答案。

class Solution {public int maxSubarraySumCircular(int[] nums) {int n = nums.length;int preMax = nums[0], maxRes = nums[0];int preMin = nums[0], minRes = nums[0];int sum = nums[0];for (int i = 1; i < n; i++) {preMax = Math.max(preMax + nums[i], nums[i]);maxRes = Math.max(maxRes, preMax);preMin = Math.min(preMin + nums[i], nums[i]);minRes = Math.min(minRes, preMin);sum += nums[i];}if (maxRes < 0) {return maxRes;} else {return Math.max(maxRes, sum - minRes);}}
}

时间复杂度: O(n),其中nnums的长度。
空间复杂度: O(1)。过程中只是用到了常数个变量。

【3】单调队列: 我们可以将数组延长一倍,即对于i≥n的元素,令nums[i]=nums[i−n]

然后,对于第二种情况,nums[0:i]nums[j:n]可以组成成连续的一段:

因此,问题转换为了在一个长度为2n的数组上,寻找长度不超过n的最大子数组和。

我们令si=∑i=0inums[i]为前缀和,如果不规定子数组的长度,只需找到最大的si−sj​,其中j<i

现在,我们只能考虑所有满足i−n≤j<ij,用单调队列维护该集合。具体的:
1、遍历到i时,单调队列头部元素下标若小于i−n,则出队。该过程一直进行,直至队列为空或者队头下标大于等于i−n
2、取队头元素作为j,计算si−sj​,并更新答案。
3、若队列尾部元素k满足sk≥si​,则出队,该过程一直进行,直至队列为空或者条件不被满足。因为k<ik更容易被步骤1剔出,并且作为被减项,sksi更大,更不具有优势。综上si要全面优于sk​。

class Solution {public int maxSubarraySumCircular(int[] nums) {int n = nums.length;Deque<int[]> queue = new ArrayDeque<int[]>();int pre = nums[0], res = nums[0];queue.offerLast(new int[]{0, pre});for (int i = 1; i < 2 * n; i++) {while (!queue.isEmpty() && queue.peekFirst()[0] < i - n) {queue.pollFirst();}pre += nums[i % n];res = Math.max(res, pre - queue.peekFirst()[1]);while (!queue.isEmpty() && queue.peekLast()[1] >= pre) {queue.pollLast();}queue.offerLast(new int[]{i, pre});}return res;}
}

时间复杂度: O(n),其中nnums的长度。我们遍历2n个元素,每个元素最多入队出队一次,因此总的时间复杂度为O(n)
空间复杂度: O(n),其中nnums的长度。

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

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

相关文章

AdaBoost算法

Boosting是一种集成学习方法&#xff0c;AdaBoost是Boosting算法中的一种具体实现。 Boosting方法的核心思想在于将多个弱分类器组合成一个强分类器。这些弱分类器通常是简单的模型&#xff0c;比如决策树&#xff0c;它们在训练过程中的错误会被后续的弱分类器所修正。Boosti…

Codeforces Round 923 (Div. 3)补题

Make it White&#xff08;Problem - A - Codeforces&#xff09; 题目大意&#xff1a;有一排格子&#xff0c;其中有黑色和白色两种&#xff0c;我们选择一个区间&#xff0c;将区间中的格子全部染成白色&#xff0c;只能选一次&#xff0c;问将这一排格子都染成白色&#x…

【doghead】VS2022 win11 安装配置WSL2 以编译linux端的cmake项目并运行1

Visual Studio 2022 在Windows上编译调试WSL2 CMake Linux工程 好像是我自己的vs2022的一个插件支持rust https://github.com/kitamstudios/rust-analyzer.vs/blob/master/PREREQUISITES.md Latest rustup (Rust Toolchain Installer). Install from here. Welcome to Rust!Th…

【性能最佳实践】跟着我们一起玩转查询模式与性能分析!

使用最新的驱动程序 MongoDB的官方驱动程序是由负责核心数据库开发的同一个专业团队打造的。这些驱动程序的更新通常比数据库本身更频繁&#xff0c;大概每几个月就会发布一次新版本。我们建议您尽可能使用最新版本的驱动程序&#xff0c;并在您使用的编程语言中安装可用的本地…

医学图像安全性概述

参考文献: Insights into security and privacy issues in smart healthcare systems based on medical images 下图左侧是医疗信息共享系统,右侧是计算机辅助诊疗策略: medical image sharing security (MISS)computer-aided diagnostic (CAD)CAD security (CADS)一般在信…

计算机毕业设计 | SSM超市进销存管理系统(附源码)

1&#xff0c;绪论 1.1 开发背景 世界上第一个购物中心诞生于美国纽约&#xff0c;外国人迈克尔库伦开设了第一家合作商店&#xff0c;为了更好地吸引大量客流量&#xff0c;迈克尔库伦精心设计了低价策略&#xff0c;通过大量进货把商品价格压低&#xff0c;通过商店一次性集…

Fink CDC数据同步(二)MySQL数据同步

1 开启binlog日志 2 数据准备 use bigdata; drop table if exists user;CREATE TABLE user(id INTEGER NOT NULL AUTO_INCREMENT,name VARCHAR(20) NOT NULL DEFAULT ,birth VARCHAR(20) NOT NULL DEFAULT ,gender VARCHAR(10) NOT NULL DEFAULT ,PRIMARY KEY(id) ); ALTER TA…

代码随想录算法训练营29期|day43 任务以及具体任务

第九章 动态规划 part05 1049. 最后一块石头的重量 II class Solution {public int lastStoneWeightII(int[] stones) {int sum 0;for (int i : stones) {sum i;}int target sum >> 1;//初始化dp数组int[] dp new int[target 1];for (int i 0; i < stones.lengt…

Java 将TXT文本文件转换为PDF文件

与TXT文本文件&#xff0c;PDF文件更加专业也更适合传输&#xff0c;常用于正式报告、简历、合同等场合。项目中如果有使用Java将TXT文本文件转为PDF文件的需求&#xff0c;可以查看本文中介绍的免费实现方法。 免费Java PDF库 本文介绍的方法需要用到Free Spire.PDF for Java…

【力扣】快乐数,哈希集合 + 快慢指针 + 数学

快乐数原题地址 方法一&#xff1a;哈希集合 定义函数 getNext(n) &#xff0c;返回 n 的所有位的平方和。一直执行 ngetNext(n) &#xff0c;最终只有 2 种可能&#xff1a; n 停留在 1 。无限循环且不为 1 。 证明&#xff1a;情况 1 是存在的&#xff0c;如力扣的示例一…

寒假作业-day5

1>现有无序序列数组为23,24,12,5,33,5347&#xff0c;请使用以下排序实现编程 函数1:请使用冒泡排序实现升序排序 函数2:请使用简单选择排序实现升序排序 函数3:请使用直接插入排序实现升序排序 函数4:请使用插入排序实现升序排序 代码&#xff1a; #include<stdio.h&g…

金融行业专题|证券超融合架构转型与场景探索合集(2023版)

更新内容 更新 SmartX 超融合在证券行业的覆盖范围、部署规模与应用场景。新增操作系统信创转型、Nutanix 国产化替代、网络与安全等场景实践。更多超融合金融核心生产业务场景实践&#xff0c;欢迎阅读文末电子书。 在金融行业如火如荼的数字化转型大潮中&#xff0c;传统架…