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]);
相关推荐
计算机小白一个3 小时前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^3 小时前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan
大数据追光猿5 小时前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!6 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉6 小时前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生6 小时前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴6 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯
Leuanghing6 小时前
【Leetcode】11. 盛最多水的容器
python·算法·leetcode
qy发大财6 小时前
加油站(力扣134)
算法·leetcode·职场和发展
王老师青少年编程6 小时前
【GESP C++八级考试考点详细解读】
数据结构·c++·算法·gesp·csp·信奥赛