代码随想录算法训练营第三十五天| 860.柠檬水找零,406.根据身高重建队列,452. 用最少数量的箭引爆气球

本文主要是介绍代码随想录算法训练营第三十五天| 860.柠檬水找零,406.根据身高重建队列,452. 用最少数量的箭引爆气球,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目与题解

860.柠檬水找零

题目链接:860.柠檬水找零

代码随想录题解:​​​​​​​860.柠檬水找零

视频讲解:贪心算法,看上去复杂,其实逻辑都是固定的!LeetCode:860.柠檬水找零_哔哩哔哩_bilibili

解题思路:

        从现实实际的角度出发,为了让手里的零钱尽可能的零碎,以备不时之需,需要找钱时肯定尽量能先找大的就找大的。这道题其实限制颇多,给解题降低了不少难度。

        首先,每个顾客只买一份柠檬水,意味着每个顾客每次消费的金额固定,为5美金。其次,顾客每次只会给5,10,20美金的面额,所以对应的情况是不需要找钱,找5美金,和找15美金。前两种的情况对应的找钱都是固定的,只有找15美金时可能有两种情况:找10+5或5*3。为了手里有更多5美金的零钱,优先找10+5的。代码如下

class Solution {public boolean lemonadeChange(int[] bills) {int[] changes = new int[2];for (int i = 0; i < bills.length; i++) {if (bills[i] == 5) {changes[0]++;} else if (bills[i] == 10) {if (changes[0] > 0) {changes[0]--;changes[1]++;} else {return false;}} else {if (changes[1] > 0 && changes[0] > 0) {changes[0]--;changes[1]--;} else if (changes[0] >= 3) {changes[0] -= 3;} else {return false;}}}return true;}
}

看完代码随想录之后的想法 

        这题其实就是现实的模拟题,基本上也没有什么算法可言,就是把想法写下来。由于这里只需要算5美金和10美金的两种剩余零钱情况,可以不需要用数组,换成普通变量记录,效率更高。

遇到的困难

        无

406.根据身高重建队列

题目链接:406.根据身高重建队列

代码随想录题解:​​​​​​​406.根据身高重建队列

视频讲解:贪心算法,不要两边一起贪,会顾此失彼 | LeetCode:406.根据身高重建队列_哔哩哔哩_bilibili

解题思路:

        首先根据身高进行降序排列,如果身高相同的,根据k降序排列。这样做的原因是,身高排序问题本质是一种插入排序,所以每插入一个新的,要保证原来的顺序不受影响。由于身高高的k值不受身高低的k值影响,所以排序的第一个key必然是身高。当二者身高相同时,k值越大的,肯定排的越后,所以优先插入k值小的。

        二维排序结束后,就要按顺序逐一插入队伍了,直接根据当前people[i]的k值,将people[i]插入对应的k位置即可,这样可以保证people[i]的前面不比他矮的人数就是k。        

class Solution {public int[][] reconstructQueue(int[][] people) {Arrays.sort(people, new Comparator<int[]>() {public int compare(int[] o1, int[] o2) {if(o1[0] != o2[0]){return o2[0] - o1[0];  //身高降序排列} else {return o1[1] - o2[1];  //前方需要插入人数升序排列}}});List<int[]> result = new ArrayList<>();for (int i = 0; i < people.length; i++) {result.add(people[i][1], people[i]);}return result.toArray(new int[people.length][]);}
}

看完代码随想录之后的想法 

        一开始写排序算法重写了很多次,一直没有想清楚插入排序之前的数组应该如何优先排序,所以怎么都写不对。偷看了一下随想录的排序方法,才写对compare函数。

        此外,虽然list类型可以用来进行插入排序,但其底层实现毕竟是数组,效率不高,换成链表会更好。

class Solution {public int[][] reconstructQueue(int[][] people) {// 身高从大到小排(身高相同k小的站前面)Arrays.sort(people, (a, b) -> {if (a[0] == b[0]) return a[1] - b[1];   // a - b 是升序排列,故在a[0] == b[0]的狀況下,會根據k值升序排列return b[0] - a[0];   //b - a 是降序排列,在a[0] != b[0],的狀況會根據h值降序排列});LinkedList<int[]> que = new LinkedList<>();for (int[] p : people) {que.add(p[1],p);   //Linkedlist.add(index, value),會將value插入到指定index裡。}return que.toArray(new int[people.length][]);}

遇到的困难

        对于多维度的贪心,还是要一个一个维度想清楚具体的贪心方法,再结合起来,不然很容易想错。

452. 用最少数量的箭引爆气球

题目链接:​​​​​​​452. 用最少数量的箭引爆气球

代码随想录题解:​​​​​​​452. 用最少数量的箭引爆气球

视频讲解:贪心算法,判断重叠区间问题 | LeetCode:452.用最少数量的箭引爆气球_哔哩哔哩_bilibili

解题思路:

        题目已经提醒了,这道题本质就是求相交区间的问题。如果多个区间同时相交,就只需要一根箭。

        因此,可以先对区间进行排序,以左边界为第一关键字,右边界为第二关键字,得到排序后的区间。然后用arrowPoint记录当前相交区间的范围,初始化为第一个区间的大小,此时需要的箭数量为1。然后遍历所有的区间,如果当前区间的左边界大于arrowPoint右边界,说明区间不相交,所需箭的数量加一,arrowPoint更新为当前区间范围;否则说明当前区间与arrowPoint有相交,此时更新arrowPoint的右边界为二者右边界的更小值即可。

class Solution {public int findMinArrowShots(int[][] points) {if (points.length <= 1) return points.length;Arrays.sort(points, new Comparator<int[]>() {@Overridepublic int compare(int[] o1, int[] o2) {if (o1[0] == o2[0])return Integer.compare(o1[1], o2[1]);return Integer.compare(o1[0], o2[0]);}});int count = 1;int[] arrowPoint = points[0];for (int i = 1; i < points.length; i++) {if (points[i][0] > arrowPoint[1]) {count++;arrowPoint = points[i];} else {arrowPoint[0] = points[i][0];arrowPoint[1] = Math.min(arrowPoint[1], points[i][1]);}}return count;}
}

看完代码随想录之后的想法 

        有一些可以优化的地方:区间排序其实只需要针对左边界,右边界可以不需要;arrowPoint也可以直接用当前point代替,减少一点变量的使用。不过这样涉及到修改point数组了,个人不是特别喜欢。

/*** 时间复杂度 : O(NlogN)  排序需要 O(NlogN) 的复杂度* 空间复杂度 : O(logN) java所使用的内置函数用的是快速排序需要 logN 的空间*/
class Solution {public int findMinArrowShots(int[][] points) {// 根据气球直径的开始坐标从小到大排序// 使用Integer内置比较方法,不会溢出Arrays.sort(points, (a, b) -> Integer.compare(a[0], b[0]));int count = 1;  // points 不为空至少需要一支箭for (int i = 1; i < points.length; i++) {if (points[i][0] > points[i - 1][1]) {  // 气球i和气球i-1不挨着,注意这里不是>=count++; // 需要一支箭} else {  // 气球i和气球i-1挨着points[i][1] = Math.min(points[i][1], points[i - 1][1]); // 更新重叠气球最小右边界}}return count;}
}

遇到的困难

        一开始重写compare方法的时候,直接return o1[0]-o2[0]了,导致有一个测试用例的结果不对,因为相减后结果超过了int的上限,溢出了,后面才改写为Integer.compare,需要吸取一下教训。

今日收获

        又学了很多贪心的方法。感觉贪心的题目与现实比较接近,最好的办法是先实际模拟一遍做法,再看是否有优化空间。

这篇关于代码随想录算法训练营第三十五天| 860.柠檬水找零,406.根据身高重建队列,452. 用最少数量的箭引爆气球的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/888896

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

Redis消息队列实现异步秒杀功能

《Redis消息队列实现异步秒杀功能》在高并发场景下,为了提高秒杀业务的性能,可将部分工作交给Redis处理,并通过异步方式执行,Redis提供了多种数据结构来实现消息队列,总结三种,本文详细介绍Re... 目录1 Redis消息队列1.1 List 结构1.2 Pub/Sub 模式1.3 Stream 结

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

SpringKafka错误处理(重试机制与死信队列)

《SpringKafka错误处理(重试机制与死信队列)》SpringKafka提供了全面的错误处理机制,通过灵活的重试策略和死信队列处理,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、Spring Kafka错误处理基础二、配置重试机制三、死信队列实现四、特定异常的处理策略五

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim