力扣2391---收集垃圾的最少总时间(Java、前缀和)

2024-05-12 07:12

本文主要是介绍力扣2391---收集垃圾的最少总时间(Java、前缀和),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

题目描述:

思路描述:

代码:

        最笨的方法:

        较好的方法:


题目描述:

给你一个下标从 0 开始的字符串数组 garbage ,其中 garbage[i] 表示第 i 个房子的垃圾集合。garbage[i] 只包含字符 'M' ,'P' 和 'G' ,但可能包含多个相同字符,每个字符分别表示一单位的金属、纸和玻璃。垃圾车收拾  单位的任何一种垃圾都需要花费 1 分钟。

同时给你一个下标从 0 开始的整数数组 travel ,其中 travel[i] 是垃圾车从房子 i 行驶到房子 i + 1 需要的分钟数。

城市里总共有三辆垃圾车,分别收拾三种垃圾。每辆垃圾车都从房子 0 出发,按顺序 到达每一栋房子。但它们 不是必须 到达所有的房子。

任何时刻只有 一辆 垃圾车处在使用状态。当一辆垃圾车在行驶或者收拾垃圾的时候,另外两辆车 不能 做任何事情。

请你返回收拾完所有垃圾需要花费的 最少 总分钟数。

示例 1:

输入:garbage = ["G","P","GP","GG"], travel = [2,4,3]
输出:21
解释:
收拾纸的垃圾车:
1. 从房子 0 行驶到房子 1
2. 收拾房子 1 的纸垃圾
3. 从房子 1 行驶到房子 2
4. 收拾房子 2 的纸垃圾
收拾纸的垃圾车总共花费 8 分钟收拾完所有的纸垃圾。
收拾玻璃的垃圾车:
1. 收拾房子 0 的玻璃垃圾
2. 从房子 0 行驶到房子 1
3. 从房子 1 行驶到房子 2
4. 收拾房子 2 的玻璃垃圾
5. 从房子 2 行驶到房子 3
6. 收拾房子 3 的玻璃垃圾
收拾玻璃的垃圾车总共花费 13 分钟收拾完所有的玻璃垃圾。
由于没有金属垃圾,收拾金属的垃圾车不需要花费任何时间。
所以总共花费 8 + 13 = 21 分钟收拾完所有垃圾。

示例 2:

输入:garbage = ["MMM","PGM","GP"], travel = [3,10]
输出:37
解释:
收拾金属的垃圾车花费 7 分钟收拾完所有的金属垃圾。
收拾纸的垃圾车花费 15 分钟收拾完所有的纸垃圾。
收拾玻璃的垃圾车花费 15 分钟收拾完所有的玻璃垃圾。
总共花费 7 + 15 + 15 = 37 分钟收拾完所有的垃圾。

思路描述:

        依据题目的意思,我们可以得出收集垃圾的最少总时间就是让每个车不浪费多余的时间,即某一个车到达一个位置后,如果该位置以后都没有属于自己收集的垃圾类型,就不再消耗时间去遍历。

        最笨的方法就是多次遍历得到结果。

        较好的方法就是一次遍历得到结果,在计算三辆车总的时间时,对于第一部分,可直接对 garbage\textit{garbage}garbage 中所有字符串的长度求和;对于第二部分,可以求出每辆车到达每类垃圾最后一次出现的位置所消耗的时间,最后进行求和。

        在第二部分进行代码实现时,可用一个变量维护 travel的前缀和。

代码:

        最笨的方法:

public int garbageCollection(String[] garbage, int[] travel) {int ans=0;int n=garbage.length;int m=travel.length;int[] lastIndex={-1,-1,-1};int[] threeSum={0,0,0};List<Map<Character,Integer>> list=new ArrayList<>();for(int i=0;i<n;i++){if (garbage[i].contains("M")) {lastIndex[0]=i;}if (garbage[i].contains("P")) {lastIndex[1]=i;}if (garbage[i].contains("G")) {lastIndex[2]=i;}char[] charArray = garbage[i].toCharArray();int len=charArray.length;Map<Character,Integer> map=new HashMap<>();map.put('M',0);map.put('P',0);map.put('G',0);for (int j=0;j<len;j++){map.put(charArray[j],map.get(charArray[j])+1);}list.add(map);}for (int i = 0; i < 3; i++) {if(lastIndex[i]==-1){continue;} else if (lastIndex[i] == 0) {Map<Character, Integer> map = list.get(0);if(i==0){ans+=map.get('M');} else if (i==1) {ans+=map.get('P');}else{ans+=map.get('G');}continue;}for(int j=0;j<lastIndex[i];){Map<Character, Integer> map = list.get(j);if(i==0){ans+=map.get('M');} else if (i==1) {ans+=map.get('P');}else{ans+=map.get('G');}ans+=travel[j];j++;if(j==lastIndex[i]){Map<Character, Integer> map2 = list.get(j);if(i==0){ans+=map2.get('M');} else if (i==1) {ans+=map2.get('P');}else{ans+=map2.get('G');}}}}return ans;}

        较好的方法:

class Solution {public int garbageCollection(String[] garbage, int[] travel) {int ans=0;int n=garbage.length;int[] distance=new int[n];int[] lastIndex={-1,-1,-1};for(int i=0;i<n;i++){if(i>0){distance[i]=distance[i-1]+travel[i-1];}ans+=garbage[i].length();if (garbage[i].contains("M")) {lastIndex[0]=i;}if (garbage[i].contains("P")) {lastIndex[1]=i;}if (garbage[i].contains("G")) {lastIndex[2]=i;}}for(int i=0;i<3;i++){if(lastIndex[i]==-1){continue;}else{ans+=distance[lastIndex[i]];}}return ans;}
}

这篇关于力扣2391---收集垃圾的最少总时间(Java、前缀和)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

服务器集群同步时间手记

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

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory