【力扣时间】【1705】【中等】吃苹果的最大数目

2023-11-03 12:50

本文主要是介绍【力扣时间】【1705】【中等】吃苹果的最大数目,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

吃苹果的最大数目

    • 1、来看题
    • 2、审题
    • 3、思路
    • 4、挽起袖子撸代码!
    • 5、解读
    • 6、提交
    • 7、咀嚼
    • 8、学习大牛!
    • 9、总结

什么苹果烂得这么快?

1、来看题

点我!

今天的题,只能说不愧是中等题吧。
从题目描述到难度都是中规中矩。

一点点来看吧。

2、审题

题目将了一个小故事,但也很清楚地表述了题意。
一句话说,这是一个贪心问题。

来分析下重点:

  1. 你会在第i天得到apples[i]个苹果,且这些苹果,将在days[i]后腐烂
  2. 你每天最多只出一个苹果。(烂了都不多吃,也不给别人)
  3. 尽管是同一颗树上结的果,但每次结果的数量和腐烂时间都毫无关联。
  4. 尽管apples[]days[]的长度都为n,但并不代表n天后就结束了,只要苹果还有剩余,你就可以继续吃下去。
  5. 可能会出现有几天没有苹果可吃的情况。

很明显的贪心问题了。
苹果的腐烂时间便是其优先级,而我们需要做得就是得出最优结果,能够吃掉更多的苹果。

3、思路

保持一贯的贪心思想,很明显苹果的腐烂时间决定了其优先级。
辣么带入现实考虑,为了尽可能多吃苹果,我们会优先吃临近腐烂的苹果。

而这也是我解题的思路。

由于days[]给出的是保质期长度,而我们需要记住的是腐烂时间,即第i天的苹果,会在第i+days[i]天腐烂。我们以这个时间来标记苹果,为它们分批。
注意,由于苹果每次的保质期长度都不同,所以可能出现后结的苹果先腐烂掉,或者不同日期的苹果在同一天腐烂的情况。

然后,我们就需要遍历apples[],记录不通批次的苹果腐烂的时间,并同时开吃!即一边遍历,我们一遍处理吃苹果的逻辑。毕竟case就已经给出期间若干天没有苹果可吃的情况。

我们以优先级队列存储不同批次的苹果,并且每次开吃时,都从最近将要腐烂的批次中取苹果来吃。保证了贪心的思路。

辣么最后,需要注意的就是遍历完apples[],即已经没有增量数据后,处理残量逻辑的情况。

思路明确!开动!

4、挽起袖子撸代码!

class Solution {public int eatenApples(int[] apples, int[] days) {int n = apples.length;int eaten = 0;PriorityQueue<int[]> rotted = new PriorityQueue<>(Comparator.comparingInt(o -> o[0]));for (int i = 0; i < apples.length; i++) {//当第i天结出的苹果数不为0时if (apples[i] != 0) {//计算腐烂的日期int rottedDay = i + days[i];rotted.offer(new int[]{rottedDay, apples[i]});}//如果有苹果可以吃if (eaten(rotted, i)) {++eaten;}}//时间来到等n天int day = n;while (!rotted.isEmpty()) {//取出最近一批将要腐烂的苹果int[] apple = rotted.poll();if (apple[0] <= day) {continue;}//能够吃掉的数量为 库存数 和 腐烂前日期差 中的更小值int ate = Math.min(apple[0] - day, apple[1]);eaten += ate;//时间也移动吃掉数量的天数day += ate;}return eaten;}/*** 恰一个** @param rotted* @param day* @return*/private boolean eaten(PriorityQueue<int[]> rotted, int day) {while (!rotted.isEmpty()) {//从队列中拿出最近一批腐烂的苹果int[] apples = rotted.peek();//如果苹果已腐烂,则继续拿取if (apples[0] <= day) {//清除此批次的所有苹果rotted.poll();continue;}//让此批次的苹果数-1--apples[1];if (apples[1] == 0) {//如果这批苹果已全部吃完,则清除数据rotted.poll();}return true;}return false;}
}

5、解读

可以看到,我使用了一个优先级队列,其元素为一个二元数组,int[0]用于记录该批苹果的腐烂日期,而int[1]则是该批苹果的数量。

除了主方法外,我还实现了一个名为eaten()的方法来处理吃的逻辑。
其逻辑便是每次从优先级队列rotted中取出最近快要过期的批次的苹果,并时其库存-1。
如果这批苹果已吃完,则将该数组出队列。

//让此批次的苹果数-1
--apples[1];if (apples[1] == 0) {//如果这批苹果已全部吃完,则清除数据rotted.poll();}

传入的day用来判断此刻已经到了第几天,来淘汰已经腐烂了的苹果批次。

//如果苹果已腐烂,则继续拿取
if (apples[0] <= day) {//清除此批次的所有苹果rotted.poll();continue;}

如果有苹果可吃,则返回true。没有时返回false。

主函数可分为两个部分。
第一部分时日期<n,我们一边处理增量的数据,一边开吃。

for (int i = 0; i < apples.length; i++) {//当第i天结出的苹果数不为0时if (apples[i] != 0) {//计算腐烂的日期int rottedDay = i + days[i];rotted.offer(new int[]{rottedDay, apples[i]});}//如果有苹果可以吃if (eaten(rotted, i)) {++eaten;}}

eaten记录了吃掉的苹果总数。

第二部分是日期>=n时,已经没有增量数据了,我们开始处理剩余的苹果数量。我当然仍然可以使用eaten(),但我嫌这样太慢,于是换了一种方法。
我选择直接出队最近一批将要腐烂的苹果,并根据当前日期判断能够吃掉多少。
能吃掉的数量为 这批苹果的剩余数量当前日期到腐烂日期间的差值中的最小值。

//能够吃掉的数量为 库存数 和 腐烂前日期差 中的更小值
int ate = Math.min(apple[0] - day, apple[1]);
eaten += ate;

毕竟可能在你吃完前,这批苹果就已经烂光了。也可能在烂光前,你已经吃完了。
注意时间的流动也是你吃掉的数量,毕竟每天仅吃一个苹果嘛。

//时间也移动吃掉数量的天数
day += ate;

6、提交

在这里插入图片描述
最后的结果如上,没想到又成了内存消耗排名较高的解法了。

7、咀嚼

en……
每次遍历并插入优先级队列,时间复杂度为O(NlongN)
之后清空队列,复杂度为O(N)
所以整体是O(NlongN)
空间复杂度为O(N)

8、学习大牛!

我看我的耗时排名不高,本以为又有黑科技了。
但大牛们的解法思路基本上都与我相同,估计又是在什么步骤浪费了些许时间吧。

总之, 学我这套就行!

9、总结

贪心复习,今天题比之之前的辣道贪心题要简单不少。废话,之前的可是困难!

下班下班!

这篇关于【力扣时间】【1705】【中等】吃苹果的最大数目的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

如何用Java结合经纬度位置计算目标点的日出日落时间详解

《如何用Java结合经纬度位置计算目标点的日出日落时间详解》这篇文章主详细讲解了如何基于目标点的经纬度计算日出日落时间,提供了在线API和Java库两种计算方法,并通过实际案例展示了其应用,需要的朋友... 目录前言一、应用示例1、天安门升旗时间2、湖南省日出日落信息二、Java日出日落计算1、在线API2

如何提高Redis服务器的最大打开文件数限制

《如何提高Redis服务器的最大打开文件数限制》文章讨论了如何提高Redis服务器的最大打开文件数限制,以支持高并发服务,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录如何提高Redis服务器的最大打开文件数限制问题诊断解决步骤1. 修改系统级别的限制2. 为Redis进程特别设置限制

如何使用 Bash 脚本中的time命令来统计命令执行时间(中英双语)

《如何使用Bash脚本中的time命令来统计命令执行时间(中英双语)》本文介绍了如何在Bash脚本中使用`time`命令来测量命令执行时间,包括`real`、`user`和`sys`三个时间指标,... 使用 Bash 脚本中的 time 命令来统计命令执行时间在日常的开发和运维过程中,性能监控和优化是不

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

Java将时间戳转换为Date对象的方法小结

《Java将时间戳转换为Date对象的方法小结》在Java编程中,处理日期和时间是一个常见需求,特别是在处理网络通信或者数据库操作时,本文主要为大家整理了Java中将时间戳转换为Date对象的方法... 目录1. 理解时间戳2. Date 类的构造函数3. 转换示例4. 处理可能的异常5. 考虑时区问题6.

服务器集群同步时间手记

1.时间服务器配置(必须root用户) (1)检查ntp是否安装 [root@node1 桌面]# rpm -qa|grep ntpntp-4.2.6p5-10.el6.centos.x86_64fontpackages-filesystem-1.41-1.1.el6.noarchntpdate-4.2.6p5-10.el6.centos.x86_64 (2)修改ntp配置文件 [r

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

poj 3723 kruscal,反边取最大生成树。

题意: 需要征募女兵N人,男兵M人。 每征募一个人需要花费10000美元,但是如果已经招募的人中有一些关系亲密的人,那么可以少花一些钱。 给出若干的男女之间的1~9999之间的亲密关系度,征募某个人的费用是10000 - (已经征募的人中和自己的亲密度的最大值)。 要求通过适当的招募顺序使得征募所有人的费用最小。 解析: 先设想无向图,在征募某个人a时,如果使用了a和b之间的关系

poj 3258 二分最小值最大

题意: 有一些石头排成一条线,第一个和最后一个不能去掉。 其余的共可以去掉m块,要使去掉后石头间距的最小值最大。 解析: 二分石头,最小值最大。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <c

poj 2175 最小费用最大流TLE

题意: 一条街上有n个大楼,坐标为xi,yi,bi个人在里面工作。 然后防空洞的坐标为pj,qj,可以容纳cj个人。 从大楼i中的人到防空洞j去避难所需的时间为 abs(xi - pi) + (yi - qi) + 1。 现在设计了一个避难计划,指定从大楼i到防空洞j避难的人数 eij。 判断如果按照原计划进行,所有人避难所用的时间总和是不是最小的。 若是,输出“OPETIMAL",若