贪心算法|53.最大子序和

力扣题目链接

cpp 复制代码
class Solution {
public:
    int maxSubArray(vector<int>& nums) {
       int result = INT32_MIN;
       int count = 0;
       for (int i = 0; i < nums.size(); i++) {
        count += nums[i];
        if (count > result) {
            result = count;
        }
        if (count <= 0) count = 0;
       }
       return result;
    }
};

这题的暴力解法很好理解,以上是贪心解法的代码。

贪心解法

贪心贪的是哪里呢?

如果 -2 1 在一起,计算起点的时候,一定是从 1 开始计算,因为负数只会拉低总和,这就是贪心贪的地方!

局部最优:当前"连续和"为负数的时候立刻放弃,从下一个元素重新计算"连续和",因为负数加上下一个元素 "连续和"只会越来越小。

全局最优:选取最大"连续和"

局部最优的情况下,并记录最大的"连续和",可以推出全局最优

从代码角度上来讲:遍历 nums,从头开始用 count 累积,如果 count 一旦加上 nums[i]变为负数,那么就应该从 nums[i+1]开始从 0 累积 count 了,因为已经变为负数的 count,只会拖累总和。

这相当于是暴力解法中的不断调整最大子序和区间的起始位置

那有同学问了,区间终止位置不用调整么? 如何才能得到最大"连续和"呢?

区间的终止位置,其实就是如果 count 取到最大值了,及时记录下来了。例如如下代码:

复制代码
if (count > result) result = count;

这样相当于是用 result 记录最大子序和区间和(变相的算是调整了终止位置)

如动画所示:

红色的起始位置就是贪心每次取 count 为正数的时候,开始一个区间的统计。

自己的思路:

贪心算法的关键代码在for循环里面

只有count是正数时才继续进行

当count小于0时,count=0,这里也是理解的关键

复制代码
if (count <= 0) count = 0; // 相当于重置最大子序起始位置,因为遇到负数一定是拉低总和

很好的理解咯,独自敲代码,没有出一点错误~

相关推荐
m0_748248026 分钟前
揭开 C++ vector 底层面纱:从三指针模型到手写完整实现
开发语言·c++·算法
七夜zippoe15 分钟前
Ascend C流与任务管理实战:构建高效的异步计算管道
服务器·网络·算法
Greedy Alg32 分钟前
LeetCode 208. 实现 Trie (前缀树)
算法
Kt&Rs33 分钟前
11.5 LeetCode 题目汇总与解题思路
数据结构·算法·leetcode
还是码字踏实35 分钟前
基础数据结构之数组的前缀和技巧:和为K的子数组(LeetCode 560 中等题)
算法·leetcode·前缀和·哈希字典
沙威玛_LHE5 小时前
树和二叉树
数据结构·算法
py有趣6 小时前
LeetCode算法学习之两数之和 II - 输入有序数组
学习·算法·leetcode
夏鹏今天学习了吗6 小时前
【LeetCode热题100(62/100)】搜索二维矩阵
算法·leetcode·矩阵
吃着火锅x唱着歌8 小时前
LeetCode 1128.等价多米诺骨牌对的数量
算法·leetcode·职场和发展
十八岁讨厌编程9 小时前
【算法训练营 · 补充】LeetCode Hot100(中)
算法·leetcode