LeetCode 1705.吃苹果的最大数目:贪心(优先队列) - 清晰题解

【LetMeFly】1705.吃苹果的最大数目:贪心(优先队列) - 清晰题解

力扣题目链接:https://leetcode.cn/problems/maximum-number-of-eaten-apples/

有一棵特殊的苹果树,一连 n 天,每天都可以长出若干个苹果。在第 i 天,树上会长出 apples[i] 个苹果,这些苹果将会在 days[i] 天后(也就是说,第 i + days[i] 天时)腐烂,变得无法食用。也可能有那么几天,树上不会长出新的苹果,此时用 apples[i] == 0days[i] == 0 表示。

你打算每天 最多 吃一个苹果来保证营养均衡。注意,你可以在这 n 天之后继续吃苹果。

给你两个长度为 n 的整数数组 daysapples ,返回你可以吃掉的苹果的最大数目

 

示例 1:

输入:apples = [1,2,3,5,2], days = [3,2,1,4,2]
输出:7
解释:你可以吃掉 7 个苹果:
- 第一天,你吃掉第一天长出来的苹果。
- 第二天,你吃掉一个第二天长出来的苹果。
- 第三天,你吃掉一个第二天长出来的苹果。过了这一天,第三天长出来的苹果就已经腐烂了。
- 第四天到第七天,你吃的都是第四天长出来的苹果。

示例 2:

输入:apples = [3,0,0,0,0,2], days = [3,0,0,0,0,2]
输出:5
解释:你可以吃掉 5 个苹果:
- 第一天到第三天,你吃的都是第一天长出来的苹果。
- 第四天和第五天不吃苹果。
- 第六天和第七天,你吃的都是第六天长出来的苹果。

 

提示:

  • apples.length == n
  • days.length == n
  • 1 <= n <= 2 * 104
  • 0 <= apples[i], days[i] <= 2 * 104
  • 只有在 apples[i] = 0 时,days[i] = 0 才成立

解题方法:贪心

思路很简单:

有一些苹果,你先吃哪个?答案是先吃腐烂最早的。

因此可以使用优先队列,从第0天开始模拟到第 n − 1 n-1 n1天:

每天将当天新苹果入队;腐烂较早的最优先;

队列不空今天还没吃到苹果时不断出队,若已腐烂则丢弃,若未腐烂则吃一个并将剩下的苹果入队。

遍历完 0 0 0 n − 1 n-1 n1天,不再有新苹果了,但队列中可能有旧苹果。

这次和之前有所不同,不再一天一天吃,而是几天几天连续算:

假设当前是第 d a y day day天,队首苹果有 a p p l e N u m appleNum appleNum个且将在 r o t D a y rotDay rotDay腐烂,则能连续吃 m a x ( 0 , m i n ( r o t D a y − d a y , a p p l e N u m ) ) max(0, min(rotDay - day, appleNum)) max(0,min(rotDayday,appleNum))天。

为什么 0 0 0 n − 1 n-1 n1天要一天一天地吃,而从第 n n n天开始可以连续吃同一批苹果?

因为从第 n n n天开始不会产生新苹果了,我们不再关心苹果的“产生日期”,只要没过保质期就能吃。

但是前 n n n天可不一样,例如第 0 0 0天产 2 2 2 1000 1000 1000天保质期的苹果,第 1 1 1天产 1 1 1个保质期 1 1 1天的苹果,那么就不能照着第 0 0 0天的 2 2 2个连续吃完,而应该第 0 0 0 3 3 3天吃第 0 0 0天产的,第 1 1 1天吃第 1 1 1天产的。

本质上来说就是前 n n n天每天都有新苹果产生,可能导致队首苹果不再是腐烂最早的苹果。

  • 时间复杂度 O ( n log ⁡ n ) O(n\log n) O(nlogn)
  • 空间复杂度 O ( n ) O(n) O(n)

AC代码

C++
/** @Author: LetMeFly* @Date: 2024-12-24 10:56:16* @LastEditors: LetMeFly.xyz* @LastEditTime: 2024-12-24 18:41:09*/
// 不能只按照腐烂截止时间
// 例如不能为了吃截止时间为6的4,5而空出第3天
/*
[1,2,3,5,2]
[3,2,1,4,2]<0-3, 1>, <1-3, 2>, <2-3, 1>, <3-7, 5>, <4-6, 2>0        1, 2                3, 6     4, 5
*/
// 前n天应该 边遍历边入队
class Solution {
public:int eatenApples(vector<int>& apples, vector<int>& days) {priority_queue<pair<int, int>, vector<pair<int, int>>, greater<>> pq;int ans = 0, day = 0;for (; day < apples.size(); day++) {if (apples[day]) {pq.push({day + days[day], apples[day]});}while (pq.size()) {auto [rotDay, appleNum] = pq.top();pq.pop();if (rotDay <= day) {continue;}ans++;appleNum--;if (appleNum) {pq.push({rotDay, appleNum});}break;}}while (pq.size()) {auto [rotDay, appleNum] = pq.top();pq.pop();int eat = max(0, min(rotDay - day, appleNum));ans += eat;day += eat;}return ans;}
};
Python
'''
Author: LetMeFly
Date: 2024-12-24 18:46:52
LastEditors: LetMeFly.xyz
LastEditTime: 2024-12-24 18:56:55
'''
from typing import List
import heapqclass Solution:def eatenApples(self, apples: List[int], days: List[int]) -> int:pq = []ans = 0for day in range(len(apples)):if apples[day]:heapq.heappush(pq, (day + days[day], apples[day]))while pq:rotDay, appleNum = heapq.heappop(pq)if rotDay <= day:continueans += 1appleNum -= 1if appleNum:heapq.heappush(pq, (rotDay, appleNum))breakday += 1  # 注意这里和C不一样,python执行完day是能取到的右值while pq:rotDay, appleNum = heapq.heappop(pq)eat = max(0, min(rotDay - day, appleNum))ans += eatday += eatreturn ans
Java
/** @Author: LetMeFly* @Date: 2024-12-24 18:58:06* @LastEditors: LetMeFly.xyz* @LastEditTime: 2024-12-24 19:07:07*/
import java.util.PriorityQueue;class Solution {public int eatenApples(int[] apples, int[] days) {PriorityQueue<int[]> pq = new PriorityQueue<int[]>((a, b) -> a[0] - b[0]);int ans = 0, day = 0;for (; day < apples.length; day++) {if (apples[day] > 0) {pq.add(new int[]{day + days[day], apples[day]});}while (pq.size() > 0) {int[] temp = pq.poll();int rotDay = temp[0], appleNum = temp[1];if (rotDay <= day) {continue;}ans++;appleNum--;if (appleNum > 0) {pq.add(new int[]{rotDay, appleNum});}break;}}while (!pq.isEmpty()) {int[] temp = pq.poll();int rotDay = temp[0], appleNum = temp[1];int eat = Math.max(0, Math.min(rotDay - day, appleNum));ans += eat;day += eat;}return ans;}
}
Go
/** @Author: LetMeFly* @Date: 2024-12-24 19:07:32* @LastEditors: LetMeFly.xyz* @LastEditTime: 2024-12-24 23:43:52*/
package main
import "container/heap"type pair struct{ rotDay, appleNum int}
type PQ []pairfunc (pq PQ) Len() int           { return len(pq) }
func (pq PQ) Less(i, j int) bool { return pq[i].rotDay < pq[j].rotDay }
func (pq PQ) Swap(i, j int)      { pq[i], pq[j] = pq[j], pq[i] }
func (pq *PQ) Push(v any)        { *pq = append(*pq, v.(pair)) }
func (pq *PQ) Pop() any          { v := (*pq)[len(*pq) - 1]; *pq = (*pq)[:len(*pq) - 1]; return v }func min_eatenApples(a, b int) int { if a < b { return a}; return b}
func max_eatenApples(a, b int) int { if a > b { return a}; return b}func eatenApples(apples []int, days []int) (ans int) {var pq PQfor day := range apples {if apples[day] > 0 {heap.Push(&pq, pair{day + days[day], apples[day]})}for len(pq) > 0 {v := heap.Pop(&pq).(pair)rotDay, appleNum := v.rotDay, v.appleNumif rotDay <= day {continue}ans++appleNum--if appleNum > 0 {heap.Push(&pq, pair{rotDay, appleNum})}break}}day := len(apples)for len(pq) > 0 {v := heap.Pop(&pq).(pair)rotDay, appleNum := v.rotDay, v.appleNumeat := max_eatenApples(0, min_eatenApples(rotDay - day, appleNum))ans += eatday += eat}return
}

Golang的堆是一个接口,需要实现Len、Less、Swap、Push、Pop方法。

可参考与ChatGPT的对话

同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~

Tisfy:https://letmefly.blog.csdn.net/article/details/144705583

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Kubernetes之NodeSelector与NodeName实战

目录 目标 版本 官网 概述 实战 NodeName实战 NodeSelector实战 目标 通过配置NodeSelector与NodeName实现Pod运行&#xff08;或优先运行&#xff09;在我们期望的节点之上。了解这两种实现方法的区别。 版本 Kubernets v1.25.0 官网 将Pod分配给节点https://kubernet…

【docker系列】打造个人私有网盘zfile

1. 介绍 是一个适用于个人的在线网盘(列目录)程序&#xff0c;可以将你各个存储类型的存储源&#xff0c;统一到一个网页中查看、预览、维护&#xff0c;再也不用去登录各种各样的网页登录后管理文件 2. 需要环境 2.1 硬件需求 CPU&#xff1a;至少1核 内存&#xff1a;推荐…

系统思考—冰山模型

“卓越不是因机遇而生&#xff0c;而是智慧的选择与用心的承诺。”—— 亚里士多德 卓越&#xff0c;从来不是一次性行为&#xff0c;而是一种习惯。正如我们在日常辅导中常提醒自己&#xff1a;行为的背后&#xff0c;隐藏着选择的逻辑&#xff0c;而选择的根源&#xff0c;源…

麒麟信安参展南京软博会,支持信创PC的新一代云桌面及全行业解决方案备受瞩目

12月20日至22日&#xff0c;由中国软件行业协会、江苏省软件行业协会等单位联合主办的2024中国&#xff08;南京&#xff09;软件产业博览会在南京国际博览中心隆重开幕。本届博览会以“软件驱动未来&#xff0c;数字闪耀金陵”为主题&#xff0c;吸引了各界目光&#xff0c;省…

【PLL】电荷泵锁相环各个环路参数意义

电荷泵锁相环(CPPLL)在模拟锁相环占据主导&#xff0c; 因为在环路中实现了积分器&#xff0c;而没有有缘放大器即&#xff1a;type 2锁相环可以使用无源RC滤波器实现&#xff0c;简化了PLL设计。 简单CPPLL 与C1串联电阻R1形成零点。 电容累积相位误差&#xff0c;提供积分路…

Java 网络原理 ①-IO多路复用 || 自定义协议 || XML || JSON

这里是Themberfue 在学习完简单的网络编程后&#xff0c;我们将更加深入网络的学习——HTTP协议、TCP协议、UDP协议、IP协议........... IO多路复用 ✨在上一节基于 TCP 协议 编写应用层代码时&#xff0c;我们通过一个线程处理连接的申请&#xff0c;随后通过多线程或者线程…

考研互学互助系统|Java|SSM|VUE| 前后端分离

【技术栈】 1⃣️&#xff1a;架构: B/S、MVC 2⃣️&#xff1a;系统环境&#xff1a;Windowsh/Mac 3⃣️&#xff1a;开发环境&#xff1a;IDEA、JDK1.8、Maven、Mysql5.7 4⃣️&#xff1a;技术栈&#xff1a;Java、Mysql、SSM、Mybatis-Plus、VUE、jquery,html 5⃣️数据库…

SpringBoot使用外置的Servlet容器(详细步骤)

嵌入式Servlet容器&#xff1a;应用打成可执行的jar 优点&#xff1a;简单、便携&#xff1b; 缺点&#xff1a;默认不支持JSP、优化定制比较复杂.&#xff1b; 外置的Servlet容器&#xff1a;外面安装Tomcat---应用war包的方式打包&#xff1b; 操作步骤&#xff1a; 方式一&…

Unity中的LayoutGroup与LayoutElement的实战应用

在开发中遇到过一个问题&#xff0c;首先我们是在4k分辨率下开发的&#xff0c;界面要求如下 我们以第二行为例子&#xff0c;第二行有3个界面&#xff0c;其中中间的界面是比较长的 面板中使用Vertical和Horizontal排列&#xff0c;并且勾选了ControlChildSize和ChildForceEx…

反应力场的生成物、反应路径分析方法

关注 M r . m a t e r i a l , \color{Violet} \rm Mr.material\ , Mr.material , 更 \color{red}{更} 更 多 \color{blue}{多} 多 精 \color{orange}{精} 精 彩 \color{green}{彩} 彩&#xff01; 主要专栏内容包括&#xff1a; †《LAMMPS小技巧》&#xff1a; ‾ \textbf…

“自动驾驶第一股” 图森未来退市转型:改名 CreateAI、发布图生视频大模型 “Ruyi”

12 月 19 日&#xff0c;自动驾驶公司图森未来&#xff08;TuSimple&#xff09;宣布启用全新品牌 CreateAI&#xff0c;并发布多项在生成式 AI 领域的进展。 CreateAI 宣布获著名武侠 IP《金庸群侠传》正版授权&#xff0c;将开发一款大型武侠开放世界 RPG 游戏。 新的 Creat…

FreeRTOS实战——一、基于HAL库项目的FreeRTOS移植步骤

FreeRTOS实战——一、基于HAL库项目的移植步骤 文章目录 FreeRTOS实战——一、基于HAL库项目的移植步骤前言一、下载和移植FreeRTOS二、系统文件配置2.1 FreeRTOSConfig.h中添加如下3个配置&#xff1a;2.2 修改stm32f1xx_it.c 前言 废话不多说&#xff0c;在FreeRTOS基础&…

编程初学者使用 MariaDB 数据库反射生成

编程初学者使用 MariaDB 数据库反射生成 数据库反射生成&#xff0c;是动词算子式通用代码生成器提供的高级功能&#xff0c;可以利用已有的数据库&#xff0c;反射生成相应数据库的前端和后端项目。此功能自动化程度很高&#xff0c;并且支持完善的元数据和数据编辑&#xff…

yolov6算法及其改进

yolov6算法及其改进 1、YOLOV6简介2、RepVGG重参思想3、YOLOv6架构改进3.1、Backbone方面3.2、SPP改进3.3、Neck改进3.4、Head改进 4、正负样本匹配与损失函数4.1、TaskAligned样本匹配4.2、VFL Loss分类损失函数4.3、SIOU损失函数4.4、DFL损失函数 1、YOLOV6简介 YOLOv6设计主…

面试241228

面试可参考 1、cas的概念 2、AQS的概念 3、redis的数据结构 使用场景 不熟 4、redis list 扩容流程 5、dubbo 怎么进行服务注册和调用&#xff0c;6、dubbo 预热 7如何解决cos上传的安全问题kafka的高并发高吞吐的原因ES倒排索引的原理 spring的 bean的 二级缓存和三级缓存 spr…

小程序配置文件 —— 13 全局配置 - window配置

全局配置 - window配置 这里讲解根目录 app.json 中的 window 字段&#xff0c;window 字段用于设置小程序的状态栏、导航条、标题、窗口背景色&#xff1b; 状态栏&#xff1a;顶部位置&#xff0c;有网络信号、时间信息、电池信息等&#xff1b;导航条&#xff1a;有一个当…

STM32 与 AS608 指纹模块的调试与应用

前言 在嵌入式系统中&#xff0c;指纹识别作为一种生物识别技术&#xff0c;广泛应用于门禁系统、考勤机、智能锁等场景。本文将分享如何在 STM32F103C8T6 开发板上使用 AS608 指纹模块&#xff0c;实现指纹的录入和识别功能。 硬件准备 STM32F103C8T6 开发板AS608 指纹模块…

3GPP R18 MT-SDT

Rel-17 指定MO-SDT允许针对UL方向的数据包进行小数据包传输。对于DL,MT-SDT(即DL触发的小数据)可带来类似的好处,即 通过不转换到 RRC_CONNECTED来减少信令开销和UE功耗,并通过允许快速传输(小而少见的)数据包(例如用于定位)来减少延迟。 在R17中,NR_SmallData_INACTIVE的工…

小程序基础 —— 02 微信小程序账号注册

微信小程序账号注册 小程序开发与网页开发不一样&#xff0c;在开始微信小程序开发之前&#xff0c;需要访问微信公众平台&#xff0c;注册一个微信小程序账号。 有了小程序的账号以后&#xff0c;才可以开发和管理小程序&#xff0c;后续需要通过该账号进行开发信息的设置、…

两个链表的第一个公共结点

输入两个链表&#xff0c;找出它们的第一个公共结点。 当不存在公共节点时&#xff0c;返回空节点。 数据范围 链表长度 [1,2000] 保证两个链表不完全相同&#xff0c;即两链表的头结点不相同。 算法思想描述 如上图所示&#xff0c;两种链表的出现的情况只有图中2中&…