【Java】HOT100 贪心算法

目录

理论基础

一、简单贪心

LeetCode455:分发饼干

二、中等贪心

2.1 序列问题

LeetCode376:摆动序列

2.2 贪心股票问题

LeetCode121:买卖股票的最佳时机

LeetCode121:买卖股票的最佳时机ii

2.3 两个维度权衡问题

LeetCode135:分发糖果

三、较难问题

区间问题 

LeetCode55:跳跃游戏i

LeetCode55:跳跃游戏ii

LeetCode452:用最少数量的箭引爆气球

LeetCode435:无重叠区间

LeetCode763:划分字母区间

LeetCode56:合并区间

其他

LeetCode53:最大子序和


贪心算法其实就是没有什么规律可言,所以大家了解贪心算法 就了解它没有规律的本质就够了。 不用花心思去研究其规律, 没有思路就立刻看题解。基本贪心的题目 有两个极端,要不就是特简单,要不就是死活想不出来。  

学完贪心之后再去看动态规划,就会了解贪心和动规的区别。

理论基础

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

举例:有一堆钞票,你可以拿走十张,如果想达到最大的金额,你要怎么拿?指定每次拿最大的,最终结果就是拿走最大数额的钱。

每次拿最大的就是局部最优,最后拿走最大数额的钱就是推出全局最优

再举一个例子如果是 有一堆盒子,你有一个背包体积为n,如何把背包尽可能装满,如果还每次选最大的盒子,就不行了。这时候就需要动态规划。动态规划的问题在下一个系列会详细讲解。

什么时候用贪心呢?没有固定的套路。刷题或者面试的时候,手动模拟一下感觉可以局部最优推出整体最优,而且想不到反例,那么就试一试贪心

贪心算法一般分为如下四步:

  • 将问题分解为若干个子问题
  • 找出适合的贪心策略
  • 求解每一个子问题的最优解
  • 将局部最优解堆叠成全局最优解


一、简单贪心

LeetCode455:分发饼干

思路:这里的局部最优就是大饼干喂给胃口大的,充分利用饼干尺寸喂饱一个,

全局最优就是喂饱尽可能多的小孩

所以可以尝试使用贪心策略,先将饼干数组和小孩数组排序。然后从后向前遍历小孩数组,用大饼干优先满足胃口大的,并统计满足小孩数量。

class Solution {
    // 思路:优先考虑胃口,先喂饱大胃口
    public int findContentChildren(int[] g, int[] s) {
        Arrays.sort(g);
        Arrays.sort(s);
        int count = 0;
        int start = s.length - 1;
        // 遍历胃口
        for (int index = g.length - 1; index >= 0; index--) {
            if(start >= 0 && g[index] <= s[start]) {
                start--;
                count++;
            }
        }
        return count;
    }
}

二、中等贪心

2.1 序列问题

LeetCode376:摆动序列

思路:局部最优是删除连续坡度上的节点,保证这个坡度有两个局部峰值

整体最优是整个序列拥有最多的局部峰值

在实际操作上,实际连删除都不用做,只需添加数组的局部峰值即可

本题的难度在于要考虑多种情况:(curdiff代表后一个坡度,prediff代表前一个坡度

即单调有平坡、上下中间有平坡和只有两个数的情况,具体分析见代码随想录

class Solution {
    public int wiggleMaxLength(int[] nums) {
        if (nums.length <= 1) {
            return nums.length;
        }
        //当前差值
        int curDiff = 0;
        //上一个差值
        int preDiff = 0;
        int count = 1;
        for (int i = 1; i < nums.length; i++) {
            //得到当前差值
            curDiff = nums[i] - nums[i - 1];
            //如果当前差值和上一个差值为一正一负
            //等于0的情况表示初始时的preDiff
            if ((curDiff > 0 && preDiff <= 0) || (curDiff < 0 && preDiff >= 0)) {
                count++;
                preDiff = curDiff;
            }
        }
        return count;
    }
}

这道题还可以用动态规划来做,这里就不讲了。之后可以试着做一做。


2.2 贪心股票问题

LeetCode121:买卖股票的最佳时机

思路:局部最优即左界最小,在局部最优的基础上要求的全局最优即右界最大,也就是差值最大。

class Solution {
    public int maxProfit(int[] prices) {
        int max = 0;
        int low = Integer.MAX_VALUE;  //保留左侧最小值
        for (int i = 0; i < prices.length; i++) {
            low = Math.min(low, prices[i]);  // 取最左最小价格,确定左界
            max = Math.max(max, prices[i] - low); // 寻找右侧最大值
        }
        return max;
    }
}

LeetCode121:买卖股票的最佳时机ii

思路:局部最优即在区间中寻找所有的最大递增子区间,且子区间之间不能重叠

整体最优:取所有最大的递增子区间,最后利润和也最大。

由于最后只要求利润,该题可以变成只要收集每天的正利润,而无需考虑区间的左右界。

// 贪心思路
class Solution {
    public int maxProfit(int[] prices) {
        int result = 0;
        for (int i = 1; i < prices.length; i++) {
            result += Math.max(prices[i] - prices[i - 1], 0);
        }
        return result;
    }
}

2.3 两个维度权衡问题

LeetCode135:分发糖果

思路:本题需要先确定一遍后,在确定另一边,从左到右的局部最优即:只要右边评分比左边大,右边的孩子就多一个糖果。全局最优:评分高的右边孩子比左边孩子糖果更多。

本题需要先从左到右更新,再从右到左,这样才能满足相邻条件。从右到左局部最优:取candyVec[i + 1] + 1 和 candyVec[i] 最大的糖果数量,保证第i个小孩的糖果数量既大于左边的也大于右边的。全局最优:相邻的孩子中,评分高的孩子获得更多的糖果。

class Solution {
    /**
         分两个阶段
         1、起点下标1 从左往右,只要 右边 比 左边 大,右边的糖果=左边 + 1,不大于就取最小糖果数1;
         2、起点下标 ratings.length - 2 从右往左, 只要左边 比 右边 大,此时 左边的糖果应该 取本身的糖果数(符合比它左边大) 和 右边糖果数 + 1 二者的最大值,这样才符合 它比它左边的大,也比它右边大
    */
    public int candy(int[] ratings) {
        int len = ratings.length;
        int[] candyVec = new int[len];
        candyVec[0] = 1;
        for (int i = 1; i < len; i++) {
            candyVec[i] = (ratings[i] > ratings[i - 1]) ? candyVec[i - 1] + 1 : 1;
        }

        for (int i = len - 2; i >= 0; i--) {
            if (ratings[i] > ratings[i + 1]) {
                candyVec[i] = Math.max(candyVec[i], candyVec[i + 1] + 1);
            }
        }

        int ans = 0;
        for (int num : candyVec) {
            ans += num;
        }
        return ans;
    }
}

三、较难问题

区间问题 

LeetCode55:跳跃游戏i

思路:总共要跳nums.length-1步才能到达终点,局部解即取最大跳跃步数max每次循环以第一轮的max为起点再次计算max的覆盖范围

整体解即所有的最大跳跃步数范围是否能覆盖到终点

class Solution {
    public boolean canJump(int[] nums) {
        int max = 0;
        for(int i=0;i<=max;i++){    //这里注意是小于等于max,max更新等于以新一轮覆盖点为起点再次覆盖
            max = Math.max(i+nums[i],max);
            if(max>=nums.length-1)  return true;
        }
        return false;
    }
}

LeetCode55:跳跃游戏ii

思路:局部解即每次的跳转步数尽可能最大,这样跳跃次数就最小

整体解即范围覆盖到终点且跳跃次数最小

本题的关键在于什么时候对步数进行+1?用到两个变量curIndex和preIndex来确定位置

class Solution {
    public int jump(int[] nums) {
        int nextIndex = 0;
        int curIndex = 0;
        int count = 0;
        for(int i=0; i<=nums.length;i++){
            nextIndex = Math.max(nextIndex,i+nums[i]);
            if(i == curIndex){   //当走到当前的最大覆盖范围时,判断是否到终点
                count++;    //如果没到,就要跳一步
                curIndex = nextIndex;   //跳到更新过的下一坐标
                if(nextIndex >= nums.length - 1) break; //如果到了则走完了
            }
        }
        return count;
    }
}

LeetCode452:用最少数量的箭引爆气球

思路:写了435,这道题就很容易了

class Solution {
    public int findMinArrowShots(int[][] points) {
        int count = 1;
        Arrays.sort(points, (x,y)->Integer.compare(x[1],y[1]));
        int end = points[0][1];
        for(int i=1;i<points.length;i++){
            //只要右界排序,记录不重叠的情况就可以了,每次不重叠就需要多一支箭
            if(end < points[i][0]){ 
                count++;
                end = points[i][1];
            }
        }
        return count;
    }
}

LeetCode435:无重叠区间

思路:重叠区间问题通常有两种思路:要么是按左边界排序,要么是按右边界

1. 按右边界升序排列,从左向右可以记录非重叠区间的个数,总数-非重叠个数=重叠个数

非重叠只要判断end和上一区间的左区间的关系

class Solution {
    public int eraseOverlapIntervals(int[][] intervals) {
        Arrays.sort(intervals, (a,b)-> {
            return Integer.compare(a[1],b[1]);
        });
        int count = 1;  //记录非重叠区间的个数
        int end = intervals[0][1];
        for(int i = 1;i < intervals.length;i++){
            if(end <= intervals[i][0]){ //无重叠区间,count++,更新end
                count++;
                end = intervals[i][1];
            }
        }
        return intervals.length-count;
    }
}

2. 按左边界升序排序,向左到右可以记录重叠的区间个数,直接得到重叠个数就是移除个数

非重叠和重叠都要更新end,之所以count取end和右界的较小值是因为,可能会出现>=3个区间重叠的情况

class Solution {
    public int eraseOverlapIntervals(int[][] intervals) {
        Arrays.sort(intervals, (a,b)-> {
            return Integer.compare(a[0],b[0]);
        });
        int count = 0;  //记录重叠区间的个数
        int end = intervals[0][1];
        for(int i = 1;i < intervals.length;i++){
            if(end <= intervals[i][0]){ //无重叠区间,更新end
                end = intervals[i][1];
            }else{
                count++;
                end = Math.min(end, intervals[i][1]);  //重叠区间的end
            }
        }
        return count;
    }
}

LeetCode763:划分字母区间

思路:首先遍历计算每个字母的最后出现下标,然后从头开始遍历,并更新最远出现下标,当到达最后序号时进行一次分割

class Solution {
    public List<Integer> partitionLabels(String s) {
        List<Integer> list = new ArrayList<>();
        int[] record = new int[26];
        //首先记录字母最后一次出现的下标
        for(int i=0;i<s.length();i++){
            record[s.charAt(i)-'a'] = i;
        }
        int idx = 0;
        int last = -1;
        //然后当遍历下标等于最后一次的下标时,添加字符串长度
        for(int i=0;i<s.length();i++){
            idx = Math.max(idx, record[s.charAt(i)-'a']);
            if(i == idx){
                list.add(idx-last); //先添加结果到list,再更新
                last = idx;
            }
        }
        return list;
    }
}

LeetCode56:合并区间

思路:同样首先按照左边界先排序,然后判断左边界和最右边界

记得判断=到底属于那种情况

class Solution {
    public int[][] merge(int[][] intervals) {
        List<int[]> res = new LinkedList<>();
        Arrays.sort(intervals, (x,y)->Integer.compare(x[0],y[0]));
        int left = intervals[0][0];
        int right = intervals[0][1];
        for(int i = 1; i<intervals.length; i++){
            if(right >= intervals[i][0]){   //重叠,合并
                right = Math.max(right, intervals[i][1]);
            }else{  //不重叠,添加旧区间,更新left、right
                res.add(new int[]{left, right});
                left = intervals[i][0];
                right = intervals[i][1];
            }
        }
        res.add(new int[]{left, right});
        return res.toArray(new int[res.size()][2]);
    }
}

其他

LeetCode53:最大子序和

思路:连续子数组局部最优,即遇到连续和count为负数,则立刻放弃当前count,置0,相当于从当前位置重新开始计算连续子数组和。整体最优即用sum来计算循环中count的最大值。

class Solution {
    public int maxSubArray(int[] nums) {
        if (nums.length == 1){
            return nums[0];
        }
        int sum = Integer.MIN_VALUE;
        int count = 0;
        for (int i = 0; i < nums.length; i++){
            count += nums[i];
            sum = Math.max(sum, count); // 取区间累计的最大值(相当于不断确定最大子序终止位置)
            if (count <= 0){
                count = 0; // 相当于重置最大子序起始位置,因为遇到负数一定是拉低总和
            }
        }
       return sum;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/589208.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【06016传感器原理与应用】第5章 气敏传感器 期末复习自考复习

第5章 气敏传感器 能够把气体信息变成电信号的装置。 一、学习目的与要求 通过本章的学习&#xff0c;熟悉并掌握气体传感器的工作原理和硬件组成结构。熟练掌握几种电阻式尤其TiO2、SnO2气敏传感器、非电阻式气敏传感器、固体电解质传感器、接触燃烧式气敏传感器的基本原理…

karpathy makemore 3

1 Introduction 这一章&#xff0c;karpathy通过分析参数初始化对模型的收敛的影响作为一个引子&#xff0c;让我们看到神经网络这一个复杂的系统&#xff0c;对grad非常敏感&#xff0c;以及有一些技巧可以解决grad敏感的问题。 2 准备工作 2.1 dataloader import torch b…

DS:顺序表、单链表的相关OJ题训练

欢迎各位来到 Harper.Lee 的学习小世界&#xff01; 博主主页传送门&#xff1a;Harper.Lee的博客主页 想要一起进步的uu可以来后台找我交流哦&#xff01; 在DS&#xff1a;单链表的实现 和 DS&#xff1a;顺序表的实现这两篇文章中&#xff0c;我详细介绍了顺序表和单链表的…

PAT (Advanced Level) - 1047 Student List for Course

模拟 #include <iostream> #include <cstring> #include <algorithm> #include <vector> using namespace std;const int N 40010, M 2510;int n, k; string name[N]; vector<int> list[M];int main() {scanf("%d%d", &n, &…

万万没想到,延缓帕金森病进展的“玄机”竟然就在腿上?【北京仁爱堂】

帕金森病患者的腿部变化&#xff0c;其实可以反应出很多问题。例如行走的变化问题、步态的异常等问题&#xff0c;可以反应病情轻重程度。而通过保持腿部肌肉活动的状态&#xff0c;也可以使帕金森病的症状得到一定的缓解&#xff0c;甚至有助于防止病情恶化。 帕金森病腿部变…

2024年五一数学建模竞赛C题论文首发

基于随机森林的煤矿深部开采冲击地压危险预测 摘要 煤炭作为中国重要的能源和工业原料&#xff0c;其开采活动对国家经济的稳定与发展起着至关重要的作用。本文将使用题目给出的数据探索更为高效的数据分析方法和更先进的监测设备&#xff0c;以提高预警系统的准确性和可靠性…

Photoshop前言

Photoshop前言 分辨率图像格式工具界面组件 分辨率 分辨率是指单位长度内包含的像素点的数量&#xff0c;其单位通常为像素/英寸&#xff08;ppi&#xff09;&#xff0c;300ppi表示每英寸包含300个像素点。对于1英寸1英寸大小的图像&#xff0c;若分辨率为72ppi&#xff0c;则…

基于Pytorch深度学习——多层感知机

本文章来源于对李沐动手深度学习代码以及原理的理解&#xff0c;并且由于李沐老师的代码能力很强&#xff0c;以及视频中讲解代码的部分较少&#xff0c;所以这里将代码进行尽量逐行详细解释 并且由于pytorch的语法有些小伙伴可能并不熟悉&#xff0c;所以我们会采用逐行解释小…

Visio 2021 (64bit)安装教程

Visio 2021 (64bit)安装教程 ​ 通知公告 Visio 2021 (64位) 是一款流程图和图表设计工具,主要用于创建和编辑各种类型的图表和图形。它提供了一个直观的界面和丰富的功能,可以帮助用户快速绘制专业的流程图、组织结构图、网络图、平面图、数据库模型等。 具体来说,Visio 20…

Go 语言数组

Go 语言提供了数组类型的数据结构。 数组是具有相同唯一类型的一组已编号且长度固定的数据项序列&#xff0c;这种类型可以是任意的原始类型例如整型、字符串或者自定义类型。 相对于去声明 number0, number1, ..., number99 的变量&#xff0c;使用数组形式 numbers[0], num…

Spring Cloud——Circuit Breaker上篇

Spring Cloud——Circuit Breaker上篇 一、分布式系统面临的问题1.服务雪崩2.禁止服务雪崩故障 二、Circuit Breaker三、resilience4j——服务熔断和降级1.理论知识2.常用配置3.案例实战&#xff08;1&#xff09;COUNT_BASED&#xff08;计数的滑动窗口&#xff09;&#xff0…

Spring Cloud——OpenFeign

Spring Cloud——OpenFeign 一、OpenFeign能干嘛二、OpenFeign的基本使用三、Feign的高级特性1.OpenFeign超时控制2.OpenFeign重试机制3.OpenFeign之性能优化HttpClient54.OpenFeign请求和响应压缩5.OpenFeign之Feign日志打印 四、参考 OpenFeign是一个声明式的Web服务客户端。…

Android --- 消息机制与异步任务

在Android中&#xff0c;只有在UIThread(主线程)中才能直接更新界面&#xff0c; 在Android中&#xff0c;长时间的工作联网都需要在workThread(分线程)中执行 在分线程中获取服务器数据后&#xff0c;需要立即到主线程中去更新UI来显示数据&#xff0c; 所以&#xff0c;如…

程序员的五大方法论

前言&#xff1a; 最近看了一篇总结程序员学习&#xff0c;晋升方法的文章&#xff0c;颇有感想&#xff0c;决定分享给大家&#xff0c;原文地址&#xff1a;给程序员的5条学习方法论 (qq.com)https://mp.weixin.qq.com/s/xVFlF9qTf9c74Emmdm0DqA 在繁忙的工作中&#xff0c;持…

如何在postman上提交文件格式的数据

如何在postman上提交文件格式的数据 今天在写一个文件上传的功能接口时&#xff0c;想用postman进行提交&#xff0c;花了些时间才找到在postman提交文件格式的数据。记录一下吧&#xff01; 1.打开postman&#xff0c;选择POST提交方式&#xff0c;然后在Params那一行的Head…

代码随想录算法训练营DAY45|C++动态规划Part7|70.爬楼梯(进阶版)、322. 零钱兑换、279.完全平方数

文章目录 70.爬楼梯&#xff08;进阶版&#xff09;322. 零钱兑换思路CPP代码 279.完全平方数思路CPP代码 70.爬楼梯&#xff08;进阶版&#xff09; 卡码网&#xff1a;57. 爬楼梯 文章讲解&#xff1a;70.爬楼梯(进阶版) 322. 零钱兑换 力扣题目链接 文章讲解&#xff1a;322…

安装英伟达nvidia p4计算卡驱动@FreeBSD14

FreeBSD也能跑cuda AI训练拉&#xff01; 在FreeBSD安装好pytorch和飞桨cpu版本后&#xff0c;尝试安装英伟达nvidia p4计算卡驱动。毕竟全靠cpu速度太慢了&#xff0c;还是GPU快啊&#xff01;在磕磕绊绊几天后&#xff0c;终于成功成功安装好nvidia p4的cuda驱动&#xff0c…

从零开始:Django项目的创建与配置指南

title: 从零开始&#xff1a;Django项目的创建与配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 后端开发 tags: DjangoWebDevPythonORMSecurityDeploymentOptimization Django简介&#xff1a; Django是一个开源的高级Python Web框架&#xff…

The Role of Subgroup Separability in Group-Fair Medical Image Classification

文章目录 The Role of Subgroup Separability in Group-Fair Medical Image Classification摘要方法实验结果 The Role of Subgroup Separability in Group-Fair Medical Image Classification 摘要 研究人员调查了深度分类器在性能上的差异。他们发现&#xff0c;分类器将个…

PHP源码_最新在线工具箱网站系统源码

项目运行截图 源码贡献 https://githubs.xyz/boot?app41 部分数据库表 SET NAMES utf8mb4; SET FOREIGN_KEY_CHECKS 0;-- ---------------------------- -- Table structure for toolbox_category -- ---------------------------- DROP TABLE IF EXISTS toolbox_category…