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 数组),直接在遍历过程中更新最大和。
  • 动态规划: 通常需要保存所有子问题的解,并根据这些解来构造最终的答案。

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

贪心代码:

java 复制代码
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实现:

java 复制代码
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]);
相关推荐
X同学的开始24 分钟前
数据结构之二叉树遍历
数据结构
limingade1 小时前
手机实时提取SIM卡打电话的信令和声音-新的篇章(一、可行的方案探讨)
物联网·算法·智能手机·数据分析·信息与通信
AIAdvocate3 小时前
Pandas_数据结构详解
数据结构·python·pandas
jiao000014 小时前
数据结构——队列
c语言·数据结构·算法
kaneki_lh4 小时前
数据结构 - 栈
数据结构
铁匠匠匠4 小时前
从零开始学数据结构系列之第六章《排序简介》
c语言·数据结构·经验分享·笔记·学习·开源·课程设计
C-SDN花园GGbond4 小时前
【探索数据结构与算法】插入排序:原理、实现与分析(图文详解)
c语言·开发语言·数据结构·排序算法
迷迭所归处5 小时前
C++ —— 关于vector
开发语言·c++·算法
leon6255 小时前
优化算法(一)—遗传算法(Genetic Algorithm)附MATLAB程序
开发语言·算法·matlab
CV工程师小林5 小时前
【算法】BFS 系列之边权为 1 的最短路问题
数据结构·c++·算法·leetcode·宽度优先