Leetcode-day26-贪心算法

找到每个阶段的局部最优,最后找到全局最优。

举个例子:十堆钞票,每堆里面取一张,找最大金额。贪心先从每堆里面找出金额最大的,进而推出总体的最大金额。

455. 分发饼干

思路:贪心:先对两个数组排序,大尺寸的饼干尽可能去满足大胃口的小孩。

需要注意的是这里内层可以用index,就避免了双层for循环

复制代码
class Solution {
    public int findContentChildren(int[] g, int[] s) {
        //思路:外层遍历按胃口从大到小遍历每个小孩,内层看有没有能满足他的饼干
        Arrays.sort(g);
        Arrays.sort(s);
        int res = 0;
        int index=s.length-1;
        for(int i=g.length-1;i>=0;i--){
            if(index>=0&&s[index]>=g[i]){
                index--;
                res++;
            }
        }
        return res;
    }
}

376. 摆动序列

这个题可以把数组中的节点按山坡画出来,按照题意,首先把数组中元素等于1,2的情况写出来,然后分析,可以把数组中的元素按照坡度画出来,如下图,其实就是找的山峰元素和最左和最右元素(元素个数大于3)。pre>0&&cur<0||pre<0&&cur>0

但是在峰顶还可能会有平的,这时候我们统一只计算峰顶(或者峰谷最右边的那个),所以就是pre>=0&&cur<0||pre<=0&&cur>0

数组首尾两端

我们可以让最右边的默认是一个摆动,res初始化为1,然后让pre初始化为0,也就意味这最左边多一个和nums[0]相等的元素,这样就融入了我们的判断条件:pre>=0&&cur<0||pre<=0&&cur>0

单调坡度有平坡

这里可以把pre=cur放到判断条件的里面,只有当出现摆动的时候才更新。

复制代码
class Solution {
    public int wiggleMaxLength(int[] nums) {
        int res=1;
        int pre = 0;
        int cur = 0;
        if(nums.length==1){
            return 1;
        }
        for(int i=0;i<nums.length-1;i++){
            cur = nums[i+1]-nums[i];
            if(cur<0&&pre>=0||pre<=0&&cur>0){
                res++;
                pre=cur;
            }
        }
        return res;
    }
}

53. 最大子数组和

最简单的就是暴力

复制代码
class Solution {
    public int maxSubArray(int[] nums) {
        int res=Integer.MIN_VALUE;
        for(int i =0;i<nums.length;i++){
            int sum=0;
            for(int j=i;j<nums.length;j++){
                sum=sum+nums[j];
                if(sum>res){
                res=sum;
                }
            }

        }
        return res;
    }
}

贪心算法

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

当子序和负数时,替换新子序和,因为无论如何加下一个数都不如直接选下一个数成为新子序和大 当子序和正数时,直接加新数字,因为无论如何替换下一个数都不如当前子序和加数字大

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

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

复制代码
class Solution {
    public int maxSubArray(int[] nums) {
        int res=Integer.MIN_VALUE;
        int sum=0;
        for(int i =0;i<nums.length;i++){
            sum+=nums[i];
            res = sum>res?sum:res;
            if(sum<0){
                sum=0;// 相当于重置最大子序起始位置,因为遇到负数一定是拉低总和
            }
        }
        return res;
    }
}
相关推荐
Jasmine_llq16 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
爱吃rabbit的mq17 小时前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
(❁´◡`❁)Jimmy(❁´◡`❁)18 小时前
Exgcd 学习笔记
笔记·学习·算法
YYuCChi18 小时前
代码随想录算法训练营第三十七天 | 52.携带研究材料(卡码网)、518.零钱兑换||、377.组合总和IV、57.爬楼梯(卡码网)
算法·动态规划
不能隔夜的咖喱18 小时前
牛客网刷题(2)
java·开发语言·算法
VT.馒头18 小时前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript
进击的小头19 小时前
实战案例:51单片机低功耗场景下的简易滤波实现
c语言·单片机·算法·51单片机
咖丨喱20 小时前
IP校验和算法解析与实现
网络·tcp/ip·算法
罗湖老棍子20 小时前
括号配对(信息学奥赛一本通- P1572)
算法·动态规划·区间dp·字符串匹配·区间动态规划
fengfuyao98521 小时前
基于MATLAB的表面织构油润滑轴承故障频率提取(改进VMD算法)
人工智能·算法·matlab