java数据结构与算法刷题-----LeetCode135. 分发糖果

java数据结构与算法刷题目录(剑指Offer、LeetCode、ACM)-----主目录-----持续更新(进不去说明我没写完):https://blog.csdn.net/grd_java/article/details/123063846

文章目录

    • 1. 左右遍历
    • 2. 进阶:常数空间遍历,升序降序分治法

在这里插入图片描述

1. 左右遍历

解题思路:时间复杂度O( n n n),遍历两次数组,空间复杂度O( n n n),需要记录左遍历一次后的同学手里的糖
  1. 先从左到右,依次给孩子们发糖,如果当前孩子分数比前一个高,就比前一个孩子多发一块糖,否则只给一块糖
  2. 此时老师们发现当前发的糖,不满足校方的要求,所以从右到左,想用right块糖和同学们手里的糖交换
  3. 如果当前孩子分数,比右边孩子的高,就以比右边孩子多发一块的数量right试图和当前孩子交换。如果比右边孩子的低,就只发一块试图交换
  4. 孩子很聪明,如果老师给的right块糖 < 自己当前持有的,孩子就不交换
代码

在这里插入图片描述

class Solution {public int candy(int[] ratings) {int n = ratings.length;int[] left = new int[n];//先从左到右处理一次,将处理结果保存起来//注意,处理左到右时,不要统计糖果总数for (int i = 0; i < n; i++) {if (i > 0 && ratings[i] > ratings[i - 1]) {//如果当前同学分数 > 前一个同学(左边同学)left[i] = left[i - 1] + 1;//比前一个同学多分一块糖果} else {//如果< = 前一个同学,先分配一块糖left[i] = 1;}}int right = 0, ret = 0;//right是通过给当前同学几块糖来交换它手里现在持有的糖(同学很聪明,只有right比自己手里的多才换),ret是糖果总数//这里才需要统计糖果总数for (int i = n - 1; i >= 0; i--) {if (i < n - 1 && ratings[i] > ratings[i + 1]) {//如果当前同学 > 后面同学(右边同学)right++;//如果它 "连续" > 若干个右边的同学, 这若干个连续的右边的同学有几个,就给他分几块糖} else {right = 1;//当不在连续时,默认用一块糖换他手里的糖,看它换不换}//如果当前同学本身已有糖果 > right(我们现在想和它交换的糖果数量),他就不换//如果right>left[i](当前同学已有糖果),同学选择用自己已有的糖,换取right块糖ret += Math.max(left[i], right);//然后将自己持有的糖统计到ret中}return ret;}
}

2. 进阶:常数空间遍历,升序降序分治法

升序降序分治法:就是将整个数组按照顺序,分成升序序列和降序序列。不再以单个元素为单位判断,而是以一个子序列为单位进行逻辑判断处理

  1. 这种算法,整个数组的第一个元素很关键
  2. 我们必须规定数组第一个元素是算作升序还是降序
  3. 因为升序的最后一个元素是最大的,降序的第一个元素是最大的。例如[1,2,3],[3,2,1],这两个序列,第一个是升序,第二个是降序,一个最大值在最后,一个最大值在最前面。
  4. 而这种思想怎么用到这道题呢?看下面的解题思路吧。
解题思路:时间复杂度O( n n n),空间复杂度O( 1 1 1)
  1. 将孩子所持有的分数的分布,看成两种形态,升序序列和降序序列
  2. 则每一个升序序列结束,必然紧跟一个降序序列,或者后面没有序列,反过来也一样,每一个降序序列后面,如果有其它序列,必然是一个升序序列
  3. 而我们要尽可能少分配糖果。
  1. 对于升序序列,第一个孩子分1块,后面每一个人比前一个人多分一块,构成升序,等差数列(1,2,3,4,5…)
  2. 对于降序序列,最后一个孩子分1块,前面每一个比后面一个人多分一块,构成降序,等差数列(…5,4,3,2,1)
  1. 每个升序序列的最后一个孩子,分这个序列中最多的糖果,每个降序序列的第一个孩子,分这个序列的最多的糖果
  2. 所以有一个关键点,如果两个相邻的序列长度相同,并且先升序后降序,那么升序序列的最后一个孩子的糖果,会和降序序列的第一个孩子的糖果数量一样,例如1,2,3,3,2,1

此时,升序序列1,2,3最后一个孩子,可以多分一块糖,变成1,2,4. 从而整体构成满足条件的序列1,2,4,3,2,1.

  1. 特殊情况(利用程序要求的漏洞,从而分配最少得糖果):如果两个相邻的孩子分数一样,后面那个孩子直接从1颗糖果开始,从头开始构建序列。因为,它只说相邻的孩子分数高的孩子糖要更多,如果分数相同,不需要遵循这个规则,直接给后面的孩子分配1颗

关键点在于,如果相邻孩子分数一样,无论升序还是降序序列,直接从头开始构建,这样就将糖果数量控制在最低了。

第一个元素算作升序还是降序。算作升序,因为先升序后降序完美适配这道题,如果你感到困惑,看下面的例子。

[1,2,3,1,0]为例,我们要不断记录相邻两个序列的长度,并且只关注升序在前,降序在后的情况。

  1. 同学1号,先分配1块糖果。其本身默认构成升序序列(inc = 1),pre = 1(pre是前一个同学持有的糖果,现在1号同学是下一个同学的前一个同学)
  2. 同学2号,发现>=同学1号,构成升序序列,给其分配2块糖果(分数高的多分配),此时inc = 2,pre = 2;
  3. 同学3号,>=同学2号,构成升序序列,分配3颗糖果,此时inc = 3.pre = 3;
  4. 同学4号,<同学3号,构成降序序列(dec = 1).判断dec == inc = false,则遵循,每个降序序列的新同学都分配1块糖果,故4号同学1块糖,pre = 1

为什么要加dec == inc = false这个条件: 如果两个相邻的序列长度相同,并且先升序后降序,那么升序序列的最后一个孩子a的糖果,会和降序序列的第一个孩子b的糖果数量一样,例如1,2,3,3,2,1。但是a>b,此时a孩子必须+1.。假设同学a分配2颗糖(同学a是上一个升序序列最后一个同学,inc = 2,表示这个升序序列中,a前面就一个同学,分配1颗糖),然后后面同学都比它分数低,构成降序序列

  1. 同学b分配1颗糖,构成序列为2,1,是满足条件的。因为b的分数比a小。同时b构成降序序列,目前dec = 1.
  2. 同学c分数比b小,因为降序序列,新同学分1颗,此时序列为a = 2,b = 1,c = 1. 但是b比c分数高,因此b需要多分1块糖,此时序列为a = 2,b = 2,c = 1。降序序列新增一个同学,dec = 2
  3. 此时dec = inc = 2. 对应的情况是,降序序列头一个同学b,和上一个升序序列最后一个同学a的糖果相同。但是a>b.此时必须多给a分一块糖(因为a比b分数高,不能和b分一样的糖),因此a+1 = 3,从而构成序列a = 3,b = 2,c = 1,这样就满足条件了
  1. 同学5号,<同学4号,构成降序(dec = 2),新同学分配1块糖果,降序序列的其它同学就都得多分配一块,也就是4号同学多分一块,此时4号同学2块糖,5号同学1块糖。并且dex == inc =false,不用考虑对前面升序序列做操作
代码

在这里插入图片描述

class Solution {public int candy(int[] ratings) {int n = ratings.length;int ret = 1;//最终答案,初始为1,代表第一个孩子,无论大小先给他一块糖//pre是上一个孩子获取的糖。//算法的关键在于处理两个相邻的升序+降序序列,dec记录当前降序序列长度,//inc记录上一个升序序列长度,初始为1,默认第一个孩子构成升序序列int inc = 1, dec = 0, pre = 1;for (int i = 1; i < n; i++) {//从第二个孩子开始if (ratings[i] >= ratings[i - 1]) {//如果当前孩子分数>前一个,则当前构成升序序列dec = 0;//降序序列的长度归0//如果当前孩子分数 == 上一个,则重新开始构建序列,直接从1开始分配糖果//如果当前>上一个,则继续构成升序序列,pre+1给当前孩子多分配一块糖pre = ratings[i] == ratings[i - 1] ? 1 : pre + 1;ret += pre;//将给当前孩子分配的糖果计入retinc = pre;//升序序列个数正好和分配的糖的数量是一致的,因为我们是1块一块叠加分配,正好构成升序序列} else {//当前孩子分数<前一个,则开始构成降序序列dec++;//降序序列长度+1if (dec == inc) dec++;//如果当前降序序列长度 == 上一次的升序序列长度,则上一次升序序列最后一个孩子a分配的糖果 == 当前降序序列第一个孩子b的。但是a的分数>b .所以a孩子必须多分配一个糖果ret += dec;//降序序列给最后一个新同学一块糖果,前面的都得+1, +1的个数正好是dec的个数pre = 1;//降序序列的最后一个同学,都分配1颗糖果}}return ret;}
}

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

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

相关文章

【四 (6)数据可视化之 Grafana安装、页面介绍、图表配置】

目录 文章导航一、Grafana介绍[✨ 特性]二、安装和配置1、安装2、权限配置&#xff08;账户/团队/用户&#xff09;①用户管理②团队管理③账户管理④看板权限 3、首选项配置4、插件管理①数据源插件②图表插件③应用插件④插件安装方式一⑤安装方式二 三、数据源管理1、添加数…

内表-ABAP开发从入门到精通笔记

内表 概念 内表是在程序内部定义的表。是定义在内存中&#xff0c;所以运行速度会比磁盘中是实体表快很多。 内表的定义&#xff0c;可以通过type来定义&#xff0c;也可以通过变量来定义。 例如&#xff1a;先定义一个结构体&#xff0c;然后再通过结构体定义内表 先顶一个结…

合合信息扫描全能王亮相静安区3·15活动,AI扫描带来绿色消费新体验

保护消费者的合法权益&#xff0c;是全社会的共同责任。为优化消费环境、促进品质消费高地建设&#xff0c;打造安全优质和谐的消费环境&#xff0c;上海静安区消保委于3月15日举办静安区2024年“315”国际消费者权益日活动。 “激发消费活力&#xff0c;绿色低碳同行”是本次3…

蓝桥杯每日一题——棋盘

问题描述 小蓝拥有 n xn 大小的棋盘&#xff0c;一开始棋盘上全都是白子。小蓝进行了 m 次操作&#xff0c;每次操作会将棋盘上某个范围内的所有棋子的颜色取反(也就是白色棋子变为黑色&#xff0c;黑色棋子变为白色)请输出所有操作做完后棋盘上每个棋子的颜色。输入格式 输入的…

智能合约语言(eDSL)—— 使用rust实现eDSL的原理

为理解rust变成eDSL的实现原理&#xff0c;我们需要简单了解元编程与宏的概念,元编程被描述成一种计算机程序可以将代码看待成数据的能力&#xff0c;使用元编程技术编写的程序能够像普通程序在运行时更新、替换变量那样操作更新、替换代码。宏在 Rust 语言中是一种功能&#x…

3.19作业

1、思维导图 2、模拟面试题 1&#xff09;TCP通信中的三次握手和四次挥手 答&#xff1a;三次握手 客户端向服务器发送连接请求 服务器向客户端回复应答并向客户端发送连接请求 客户端回复服务端&#xff0c;并建立联系 四次挥手 进程a向进程b发送断开连接请求…

Linux 磁盘的一生

注意&#xff1a;实验环境都是使用VMware模拟 ​ 磁盘接口类型这里vm中是SCSI&#xff0c;扩展sata,ide(有时间可以看看或者磁盘的历史) ​ 总结&#xff1a;磁盘从有到无—类似于建房子到可以住 ————————————————————————————————————…

【linux】环境变量(进程二)

这里写目录标题 命令行参数&#xff1a;环境变量&#xff1a; 命令行参数&#xff1a; 不谈命令行参数就谈环境变量就是耍流氓。 相信我们在C语言阶段都在main函数里见过参数。 例如int main(int argc, char* argv[]) 这是什么东西呢&#xff1f; 话不多说我们直接打印一下看…

OSPF特殊区域(stub\nssa)

stub区域——只有1类、2类、3类&#xff1b;完全stub区域——只有1类、2类 NSSA区域&#xff1a;本区域将自己引入的外部路由发布给其他区域&#xff0c;但不需要接收其他区域的路由 在NSSA区域的路由器上&#xff0c;引入外部路由时&#xff0c;不会转换成5类LSA&#xff0c…

物资管理系统建设方案

二、 项目概述 2.1 项目背景 2.2 现状分析 2.2.1 业务现状 2.2.2 系统现状 三、 总体需求 3.1 系统范围 3.2 系统功能 3.3 用户分析 3.4 假设与依赖关系 四、 功能需求 五、 非功能性需求 5.1 用户界面需求 5.2 软硬件环境需求 5.3 产品质量需求 5.4 接口需求 …

第二门课:改善深层神经网络<超参数调试、正则化及优化>-超参数调试、Batch正则化和程序框架

文章目录 1 调试处理2 为超参数选择合适的范围3 超参数调试的实践4 归一化网络的激活函数5 将Batch Norm拟合进神经网络6 Batch Norm为什么会奏效&#xff1f;7 测试时的Batch Norm8 SoftMax回归9 训练一个SoftMax分类器10 深度学习框架11 TensorFlow 1 调试处理 需要调试的参…

《计算机考研精炼1000题》为你考研之路保驾护航

创作背景 在这个充满挑战与竞争的时代&#xff0c;每一位考生在备战研究生考试的过程中&#xff0c;都希望通过更多符合考纲要求的练习题来提高自己的知识和技能。为了满足这一需求&#xff0c;我们精心策划和编辑了这本《计算机考研精炼1000题》。在考研政治和考研数学领域&a…

教务管理系统(java+mysql+jdbc+Druid+三层架构)

1、项目要求 1.1数据库表描述 设计一个教务管理系统&#xff0c;要求如下&#xff1a; 系统涉及的表有 account表&#xff08;账号表&#xff09; teacher表&#xff08;教师表&#xff09; student表&#xff08;学生表&#xff09; course表 (课程表) score表&#xff08;成…

杰发科技AC7801——读取Flash数据做CRC校验

查看Keil的编译结果发现总共6160个字节。计算结果如下&#xff0c; 代码如下 #include "ac780x_crc.h" #include "ac780x.h" #include "ac780x_debugout.h" #include "string.h" #include "ac780x_eflash.h"#define TestSi…

麒麟 V10 一键安装 Oracle 11GR2(231017)单机版

Oracle 一键安装脚本&#xff0c;演示 麒麟 V10 一键安装 Oracle 11GR2 单机版过程&#xff08;全程无需人工干预&#xff09;&#xff1a;&#xff08;脚本包括 ORALCE PSU/OJVM 等补丁自动安装&#xff09; ⭐️ 脚本下载地址&#xff1a;Shell脚本安装Oracle数据库 脚本第…

模拟面试

1.TCP通信中的三次握手和四次挥手过程 三次握手 1.客户端像向服务器端发送连接请求 2.服务器应答连接请求 3.客户端与服务器简历连接 四次挥手&#xff1a; 客户端或服务器端发起断开请求,这里假设客户端发送断开请求 1.客户端向服务器发送断开请求 2.服务器应答断开请求 3.服…

Java面试相关问题

一.MySql篇 1优化相关问题 1.1.MySql中如何定位慢查询&#xff1f; 慢查询的概念&#xff1a;在MySQL中&#xff0c;慢查询是指执行时间超过一定阈值的SQL语句。这个阈值是由long_query_time参数设定的&#xff0c;它的默认值是10秒1。也就是说&#xff0c;如果一条SQL语句的执…

【开发环境搭建篇】IDEA安装和配置

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过大学刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0…

单片机第四季-第一课:RTOS

1&#xff0c;RTOS来龙去脉 操作系统是什么&#xff1f; 以人类社会类比&#xff0c;小公司三四个人都是干活的&#xff0c;大公司有几万人其中有几千人从事管理工作&#xff0c;他们的工作是让其他人的干活效率更高。 51单片机为什么没有操作系统&#xff0c;因为51的性能太…

黑马微服务p30踩坑

报错详情 : orderservice开不起来 : 发生报错 : 然后检查了以下端口啥的 &#xff0c;配置啥的都是没有问题的 ; 解决办法 : 1 . 修改nacos1,2,3中的端口&#xff0c;将conf 中 cluster.conf中 的 127.0.0.1 全部改成自己本机的真实ipv4地址; 本机真实ipv4地址查看 :…