news 2026/6/23 8:49:35

贪心算法专题(三):负重前行,不如从头再来——「最大子序和」

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
贪心算法专题(三):负重前行,不如从头再来——「最大子序和」

哈喽各位,我是前端小L。

欢迎来到贪心算法专题第三篇! 我们要解决的问题很简单:给你一个整数数组(有正有负),让你找出一个连续的子数组,使得其和最大。

这道题如果用暴力法(两层循环枚举所有子数组),复杂度是 O(N2),会超时。 如果用贪心(或者说动态规划的降维版),复杂度直接降为 O(N)。

力扣 53. 最大子序和

https://leetcode.cn/problems/maximum-subarray/

题目分析:

  • 输入:整数数组nums

  • 目标:找到一个具有最大和的连续子数组。

  • 输出:该子数组的和。

例子:nums = [-2, 1, -3, 4, -1, 2, 1, -5, 4]

  • 连续子数组[4, -1, 2, 1]的和是6,是最大的。

核心思维:什么时候该“放弃”?

我们需要一个变量count来记录当前累加的和。 从左向右遍历数组:

  1. 累加count += nums[i]

  2. 贪心决策

    • 如果count变成了负数(比如-2),它对于后面的元素来说,就是一个“累赘”。

    • 因为:负数 + 下一个数 < 下一个数

    • 后面的数会想:“与其加上你这个负数变小,我还不如自己另起炉灶!”

    • 策略:一旦count < 0,立刻把count重置为0(相当于放弃之前的序列,从下一个元素开始重新累加)。

注意:我们需要一个全局变量result来记录我们在过程中遇到的最大值。因为可能整个数组都是负数,或者最大和出现在中间某一段,而不是结尾。

算法流程

  1. 初始化

    • result = INT_MIN(或者nums[0]):记录历史最大值。

    • count = 0:记录当前连续子序列的和。

  2. 遍历数组

    • count += nums[i]

    • 更新最大值if (count > result) result = count。这一步要在重置之前做,确保我们记录下了这一瞬间的高光时刻。

    • 贪心重置if (count < 0) count = 0。如果当前和跌破 0,归零,重新开始。

  3. 返回result

代码实现 (C++)

C++

#include <vector> #include <climits> // for INT_MIN using namespace std; class Solution { public: int maxSubArray(vector<int>& nums) { int result = INT_MIN; // 最终结果 int count = 0; // 当前累加和 for (int i = 0; i < nums.size(); i++) { count += nums[i]; // 1. 累加 // 2. 取最大值 (必须放在重置之前,防止全负数的情况漏掉最大值) if (count > result) { result = count; } // 3. 贪心策略:如果当前和已经是负数了,就扔掉它 if (count < 0) { count = 0; } } return result; } };

深度辨析:贪心 vs 动态规划

这道题其实也是经典的DP题目。

  • DP 定义dp[i]表示以nums[i]结尾的最大连续子数组和。

  • 转移方程dp[i] = max(dp[i-1] + nums[i], nums[i])

    • 如果dp[i-1]是负的,那就不要它了,直接取nums[i]

    • 如果dp[i-1]是正的,那就加上它。

  • 观察:你看,这不就是我们的贪心逻辑吗?count其实就是dp[i-1],当它小于 0 时,我们重置为 0(相当于只取nums[i])。

贪心写法只是把 DP 数组的空间优化到了 O(1) 而已。

深度复杂度分析

  • 时间复杂度:O(N)

    • 只需要遍历一次数组。

  • 空间复杂度:O(1)

    • 只需要两个变量countresult

总结:学会“及时止损”

今天这道题的贪心哲学非常实用:不要因为沉没成本而犹豫。当你发现当前的积累(count)已经变成负资产时,无论之前付出了多少努力(遍历了多少元素),都要果断“归零”,轻装上阵,才能在未来(后面的序列)取得更大的成就(result)。

下一题预告: 如果我们在炒股,这就更刺激了。 假设你能预知未来几天的股价,并且可以无限次地买入卖出(T+0,当天买当天卖都行),你怎么操作才能赚得最多? 贪心策略简直简单到令人发指:只要今天比昨天贵,我就赚这个差价!

下期见!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/23 21:18:22

STL容器——String容器

STL容器——String容器 基本概念 string是c风格的字符串&#xff0c;不同于c语言的char*&#xff0c;本质是一个类 char*是一个指针&#xff0c;string是一个类&#xff0c;内部封装了char *来管理字符串&#xff0c;是一个char&型的容器 特点&#xff1a;内部封装了查找f…

作者头像 李华
网站建设 2026/6/23 17:34:55

详细分析一下 国富论里里面 十一章 论 地租

《国富论》第十一章《论地租》&#xff08;Of the Rent of Land&#xff09;是亚当斯密对土地收入性质的系统性分析&#xff0c;属于其分配理论的重要组成部分。本章在斯密的理论体系中具有承上启下的作用&#xff0c;既延续了对生产要素报酬&#xff08;工资、利润、地租&…

作者头像 李华
网站建设 2026/6/23 17:37:36

现在 夸脱小麦 多少 盎司白银

根据当前的国际市场价格数据&#xff0c;1夸脱小麦约相当于0.24盎司白银。这个比例与您在《国富论》中读到的历史数据&#xff08;如14世纪约4盎司白银/夸脱&#xff09;相比&#xff0c;已经发生了巨大变化。下面是根据最新市场数据进行的计算和对比分析&#xff1a;&#x1f…

作者头像 李华
网站建设 2026/6/22 20:21:01

Java Web html 图书管理系统系统源码-SpringBoot2+Vue3+MyBatis-Plus+MySQL8.0【含文档】

摘要 随着信息技术的快速发展&#xff0c;图书管理系统在图书馆、学校及企业中的应用日益广泛&#xff0c;传统的手工管理模式已无法满足高效、精准的管理需求。数字化图书管理系统能够实现图书信息的快速检索、借阅记录的自动化管理以及用户权限的精细化控制&#xff0c;极大地…

作者头像 李华
网站建设 2026/6/23 17:30:26

半光滑牛顿法非线性优化带35个测试函数 半光滑牛顿法求解非线性目标函数约束优化问题的MATLA...

半光滑牛顿法非线性优化带35个测试函数 半光滑牛顿法求解非线性目标函数约束优化问题的MATLAB自编源代码&#xff0c;不调用MATLAB优化库函数&#xff0c;每个函数开头有简单英语注释&#xff0c;求解速度比MATLAB自带优化库函数快。 目标函数支持非线性目标函数、二次型函数等…

作者头像 李华