当前位置: 首页 > news >正文

leetcode_53. 最大子数组和

53. 最大子数组和

题目描述:给你一个整数数组 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 <= 10^5
  • -10^4 <= nums[i] <= 10^4

进阶:如果你已经实现复杂度为 O(n) 的解法,尝试使用更为精妙的 分治法 求解。

贪心算法的解释:

  • 贪心选择: 在每一步中,我们都做出一个局部最优的选择,希望最终得到全局最优解。在最大子数组和问题中,每一步的贪心选择是决定是否将当前元素加入前面的子数组,还是从当前元素重新开始一个新的子数组.

这种贪心选择确保了在每一步我们都在试图构建一个尽可能大的子数组和,并且在整个遍历过程中保持对全局最优解(最大子数组和)的追踪。

动态规划的解释:

这种方法也可以看作是一种动态规划(因为我们在每一步都依赖之前的结果来做决策),但因为我们在每一步都只需要依赖上一步的状态,所以可以将这个算法实现为一个贪心算法。

  • 贪心算法: 不需要显式的状态转移表(如 DP 数组),直接在遍历过程中更新最大和。
  • 动态规划: 通常需要保存所有子问题的解,并根据这些解来构造最终的答案。

在这个问题中,贪心策略和动态规划方法是等价的。贪心的选择在每一步都能确保局部最优,并最终达到全局最优。

贪心代码:

class Solution {public int maxSubArray(int[] nums) {int max = nums[0];int count = 0;for (int num : nums) {if (count < 0) {count = num;} else {count += num;}max = Math.max(max, count);}return max;}
}

代码思路:

  • 如果当前子数组的和 count 小于 0,那么把当前元素作为新子数组的开始(因为如果继续累加一个负值,只会使和变得更小)。
  • 如果当前子数组的和 count 大于等于 0,那么就继续累加当前元素。

显式DP实现:

class Solution {public int maxSubArray(int[] nums) {// dp[i] 表示以 nums[i] 结尾的最大子数组和int[] dp = new int[nums.length];dp[0] = nums[0];int max = dp[0];// 从第二个元素开始遍历for (int i = 1; i < nums.length; i++) {// dp[i] 要么是自己,要么是自己加上前面的最大子数组和dp[i] = Math.max(nums[i], dp[i - 1] + nums[i]);// 更新全局最大子数组和max = Math.max(max, dp[i]);}return max;}
}

代码思路:

  1. 定义 DP 数组: dp[i] 表示以 nums[i] 结尾的最大子数组和
  2. 初始化dp[0] = nums[0]DP 数组的第一个元素直接等于 nums[0],因为子数组只能包含自己。int max = dp[0]; 初始化 max 为第一个元素的值。
  3. DP 状态转移: 对于每个 i (从1到 nums.length - 1):dp[i] = Math.max(nums[i], dp[i - 1] + nums[i]);
    1. 如果 dp[i-1] + nums[i] 是负的,从当前元素重新开始一个新的子数组。
    2. 如果前一个子数组的和 dp[i-1] 加上当前元素 nums[i] 是正的,说明可以扩展当前子数组并获得更大的和。
  4. 更新最大值:在每一步都更新全局最大子数组和 max = Math.max(max, dp[i]);

http://www.mrgr.cn/news/470.html

相关文章:

  • MCU复位RAM会保持吗,如何实现复位时变量数据保持
  • 网络编程 8/15 基于UDP多人聊天室
  • linux部署elasticserch单节点
  • js取消焦点事件
  • 【边缘计算与智能家居】边缘计算在智能家居中的应用
  • c#实现数据导出为PDF的方式
  • Java用JNA调用dll : Invalid memory access
  • 稚晖君发布5款全能人形机器人,开源创新,全能应用
  • 一元闯关游戏
  • 宝塔面板部署webman项目+nginx反向代理
  • 深度学习--转换拼接问题 + TensorFlow包弃用问题
  • 在MuMu模拟器中的游戏如何设置变声器?电脑变声器开麦就变声!6款实时变声软件!TM真的炫!
  • SQLite3使用接口写入二进制文件
  • 高级java每日一道面试题-2024年8月15日-设计模式篇-设计模式与面向对象原则的关系是什么?
  • 构筑信息安全的桥梁:安全信息交换(SIX)全面解析
  • C++之函数传参方式
  • ImageMagick从pdf导出高清图片
  • 宝兰德荣获openEuler项目群青铜捐赠人称号,共筑开源生态繁荣新篇章
  • STM32标准库学习笔记-9.DMA 直接存储器存取
  • 二、前后端分离通用权限系统(2)