代码随想录算法训练营day27

代码随想录算法训练营

---day27

文章目录


前言

今天是算法营的第27天,希望自己能够坚持下来!

今日任务:

● 贪心算法理论基础

● 455.分发饼干

● 376. 摆动序列

● 53. 最大子序和


一、贪心算法理论基础

文章讲解
视频讲解

题目大纲:

贪心的本质是选择每一阶段的局部最优,从而达到全局最优。

贪心没有套路,说白了就是常识性推导加上举反例。贪心的题目要么很简单,要么没做过就想不出来思路。遇到没思路的题目不要想太久,马上看题解积累思路。


二、455.分发饼干

题目链接
文章讲解
视频讲解

思路:

  1. 这里的局部最优是,尽量用大的饼干去满足大的胃口(或者反过来用小的饼干满足小的胃口)
  2. 先对饼干和胃口排序
  3. 从后往前遍历胃口,优先用最大的饼干去匹配大胃口
  4. 如果满足的话则饼干index往前(这里要注意index>=0),累加计数。

代码如下:

cpp 复制代码
class Solution {
public:
    int findContentChildren(vector<int>& g, vector<int>& s) {
        //先排列,升序
        sort(g.begin(), g.end());
        sort(s.begin(), s.end());

        int index = s.size() - 1;
        int result = 0;

        for (int i = g.size() - 1; i >= 0; i--) { //遍历胃口
            if (index >= 0 && s[index] >= g[i]) { //遍历饼干,用最大的饼干匹配
                index--;
                result++;
            }
        }

        return result;
    }
};

三、376. 摆动序列

题目链接
文章讲解
视频讲解

局部最优:删除单调坡度上的节点(不包括单调坡度两端的节点),那么这个坡度就可以有两个局部峰值(头和尾)。

整体最优:整个序列有最多的局部峰值,从而达到最长摆动序列。

因为题目要求的是最长摆动子序列的长度,所以只需要统计数组的峰值数量就可以了

思路:

  1. 用当前元素分别和前一元素,后一元素的差的正负来判断是否有摆动;
  2. preDiff = nums[i + 1] - nums[i],curDiff = nums[i] - nums[i - 1];
  3. preDiff和curDiff一正一负时说明出现了摆动。

这是我们思考本题的一个大体思路,但本题要考虑三种情况:

情况一:上下坡中有平坡

平坡有4个2,那么考虑删掉左边3个2的话,遍历到最后一个2时的条件就是prediff = 0 && curdiff < 0,所以判断峰值的条件就应该是:

(preDiff <= 0 && curDiff > 0) || (preDiff >= 0 && curDiff < 0)

情况二:数组首尾两端

题目中说了,如果只有两个不同的元素,那摆动序列也是 2。

那么为了统计到这种情况,默认最右端是一个摆动,result初始化为1,且遍历只遍历到倒数第二个,i < nums.size() - 1;再加上情况一讨论的判断条件 curDiff > 0 && preDiff <= 0,那么result++,就会得到答案2.

情况三:单调坡中有平坡

为了避免nums[1]和nums[2]都各自统计了一次摆动,只需要在这个坡度摆动变化的时候,更新 prediff 就行(也就是图上的1),这样 prediff 在 单调区间有平坡的时候 就不会发生变化,造成我们的误判。

cpp 复制代码
class Solution {
public:
    int wiggleMaxLength(vector<int>& nums) {
        if (nums.size() <= 1) return nums.size();

        int preDiff = 0; //nums[i + 1] - nums[i]
        int curDiff = 0; //nums[i] - nums[i - 1]
        int result = 1; //默认最右边有一个峰值

        //只遍历到倒数第二个,因为最右边一个已经算成一个峰值了
        for (int i = 0; i < nums.size() - 1; i ++) {
            curDiff = nums[i + 1] - nums[i];

            //出现峰值
            if ((preDiff <= 0 && curDiff > 0) || (preDiff>= 0 && curDiff < 0)) {
                result++;
                preDiff = curDiff; //当遇到摆动变化时才更新prediff
            }
        }

        return result;
    }
};

53. 最大子数组和

题目链接
文章讲解
视频讲解

思路:

  1. 当累加和是负数时,继续往后加都是让后面的数更加小
  2. 所以当累加和为负时,舍弃掉当前的累加,重新从下一个元素开始累加,并且在过程中记录累加的最大值。

代码如下:

cpp 复制代码
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]; //计算累加值
            if (count > result) result = count; //记录最大值
            if (count < 0) count = 0; //当连续和为负数时,舍弃累加值,重新累加
        }

        return result;
    }
};

总结

今天第一天的贪心算法,代码其实都不难,难的是思路,需要多积累一些解题思路才行。

明天继续加油!

相关推荐
就叫飞六吧30 分钟前
51 单片机和 STM32 引脚命名对照表与解析
c++·stm32·单片机·嵌入式硬件·51单片机
霜雪殇璃31 分钟前
c++对结构体的扩充以及类的介绍
开发语言·c++·笔记·学习
冉佳驹35 分钟前
C++ ——— 匿名对象
c++·学习·类和对象·匿名对象
誓约酱1 小时前
git的基本使用
linux·运维·服务器·c++·git·后端
范纹杉想快点毕业2 小时前
XML通过HTTP POST 请求发送到指定的 API 地址,进行数据回传
xml·c语言·开发语言·数据结构·c++·python·c#
float_六七2 小时前
C/C++头文件locale
c语言·开发语言·c++
bachelores2 小时前
数据结构-排序
数据结构·算法·排序算法
飞鸢逐浪2 小时前
3D形状匹配 - SpiderMatch
算法
誓约酱2 小时前
Linux下字符设备驱动编写(RK3568)
linux·运维·服务器·c语言·c++·嵌入式硬件·物联网
月亮有痕迹诶2 小时前
找到一个或多个多重定义的符号的问题
c++