【每日算法Day 69】面试经典题:分发糖果问题

简介: 分发糖果

题目链接


LeetCode 135. 分发糖果[1]

题目描述



image.png

示例1

输入:
[1,0,2]
输出:
5
解释:
你可以分别给这三个孩子分发 2、1、2 颗糖果。

示例2

输入:
[1,2,2]
输出:
4
解释
:你可以分别给这三个孩子分发 1、2、1 颗糖果。第三个孩子只得到 1 颗糖果,这已满足上述两个条件。

题解


这题虽然难度定义成困难,但其实代码不长,思路也比较简单清晰。

首先明确一下题目中的两个条件,我们可以把所有人的分数在坐标轴中连起来,这样就形成了一个波形图(图片来自官方题解):image.png

image.png

要注意的一点是图中 pt.13 那个位置也是一个谷底,因为它右边的分数等于它。

那么问题的关键就是如何找到这些谷底了。

两次遍历


首先初始化所有人都

分到  个糖果。

然后从左向右遍历一次所有分数,如果发现分数小于等于前一个人分数,那暂时不用管,因为你从左向右是没法知道下坡的点距离谷底还有多远的。而如果发现分数大于前一个人分数,那么就在前一个人糖果数基础上,再多分一个给他,因为是上坡,所以谷底一定在左边已经遍历过了,是知道距离的。

接着就是遍历下坡了,也就是从右向左遍历所有分数,同理如果发现分数大于后一个人分数,那么就在后一个人糖果数基础上,再多分一个给他。

但是这里要处理一个冲突,那就是峰顶既是上坡,又是下坡,其实只要两次遍历完取上坡和下坡中糖果数较大的那个就行了。

总结一下就是一次从左向右遍历所有上坡,一次从右向左遍历所有下坡,峰顶取两次较大值

image.png

一次遍历


从上面方法中可以看出,本题求解的难点就在于从左向右遍历的时候,下坡到底有多长没法知道,必须全部遍历完了才能知道。还有就是山峰的值必须看左右两边的上坡下坡有多长。

为了解决这个问题,我们可以用变量 up 记录上坡的长度,down 记录下坡的长度。

当遇到谷底的时候,就表明一座山遍历结束了,那么我们只需要比较 updown 的大小就知道峰顶取值了。


image.png

在具体实现的时候,这个方法细节有点多,一不小心就会错,直接看代码注释吧。

继续看下面这张图:image.png贴一段官方的样例解释:image.png

image.png

单调栈



image.png

image.png

小结


第一种解法最容易理解和实现,也不用考虑什么特殊情况。但是后两种方法处理起来就稍稍有点麻烦了,需要结合样例和代码来理解。但是本质上都是一样的,都是从谷底(糖果数为 )开始,向两周扩展。方法一是先从每个谷底向右边上坡扩展,再向左边下坡扩展。方法二是计算出相邻两个谷底之间的上坡下坡长度,然后直接计算。第三个方法是从每个谷底先向左边下坡扩展,再向右边上坡扩展。

代码


两次遍历(c++)


classSolution {
public:   
intcandy(vector<int>&ratings) {  
intn=ratings.size(); 
vector<int>res(n, 1);  
for (inti=1; i<n; ++i) {   
if (ratings[i] >ratings[i-1]) {     
res[i] =res[i-1] +1;   
            }     
        }     
for (inti=n-2; i>=0; --i) {   
if (ratings[i] >ratings[i+1]) {   
res[i] =max(res[i], res[i+1]+1); 
            }     
        }     
returnaccumulate(res.begin(), res.end(), 0); 
    }
};

一次遍历(c++)

classSolution {
public:  
intcount(intn) {    
returnn*(n+1)/2;    }
intcandy(vector<int>&ratings) {   
intn=ratings.size();     
intsum=0;     
intup=0, down=0;    
intos=0, ns=0;    
for (inti=1; i<n; ++i) {      
ns=ratings[i]>ratings[i-1] ?1 : (ratings[i]
<ratings[i-1] ?-1 : 0);     
// 这座山峰遍历结束,计算糖果数。  if ((os<0&&ns>=0) ||os>0&&ns==0) {    
// 这里看似好像峰顶没有加 1,其实是 count(down) 减去了 1。                // 因为谷底是共享的,所以将谷底给了下一座山峰的上坡。           sum+=count(up) +count(down) +max(up, down);                up=down=0;     
            }        
if (ns>0) up++;    
elseif (ns<0) down++;    
// 如果是平原,说明谷底不会共享,之前少加的 1 再补上。            else if (!ns) sum++;    os=ns;    
        }     
// 最后一座山峰循环里不会计算到,再加上。    sum+=count(up) +count(down) +max(up, down) +1;        returnsum; 
    }
};

单调栈(c++)


classSolution {
public:  
intcandy(vector<int>&ratings) {     
ratings.push_back(INT_MAX);   
intn=ratings.size();  
vector<int>res(n, 1);  
stack<int>st;   
intsum=0;     
for (inti=0; i<n; ++i) {  
if (!st.empty() &&ratings[i] >=ratings[st.top()]) {                while (!st.empty()) {        
intj=st.top();    
st.pop();      
if (j<n-1&&ratings[j] >ratings[j+1]) {    
res[j] =max(res[j], res[j+1]+1);  
                }         
if (j>0&&ratings[j] >ratings[j-1]) {   
res[j] =max(res[j], res[j-1]+1);     
                }   
sum+=res[j]; 
            }   
                                                                }            st.push(i);  
        }      
returnsum;  
    }
};

参考资料

[1]

LeetCode 135. 分发糖果: https://leetcode-cn.com/problems/candy/

image.png

作者简介:godweiyang知乎同名华东师范大学计算机系硕士在读,方向自然语言处理与深度学习喜欢与人分享技术与知识,期待与你的进一步交流~


相关文章
|
3天前
|
存储 算法 编译器
米哈游面试算法题:有效的括号
米哈游面试算法题:有效的括号
28 0
|
3天前
|
开发框架 算法 搜索推荐
C# .NET面试系列九:常见的算法
#### 1. 求质数 ```c# // 判断一个数是否为质数的方法 public static bool IsPrime(int number) { if (number < 2) { return false; } for (int i = 2; i <= Math.Sqrt(number); i++) { if (number % i == 0) { return false; } } return true; } class Progr
64 1
|
3天前
|
负载均衡 算法 应用服务中间件
面试题:Nginx有哪些负载均衡算法?Nginx位于七层网络结构中的哪一层?
字节跳动面试题:Nginx有哪些负载均衡算法?Nginx位于七层网络结构中的哪一层?
49 0
|
1天前
|
算法 前端开发 Android开发
Android文字基线Baseline算法的使用讲解,Android开发面试题
Android文字基线Baseline算法的使用讲解,Android开发面试题
Android文字基线Baseline算法的使用讲解,Android开发面试题
|
1天前
|
算法 Java API
Groovy脚本基础全攻略,android面试算法题
Groovy脚本基础全攻略,android面试算法题
|
2天前
|
NoSQL 算法 Java
【redis源码学习】持久化机制,java程序员面试算法宝典pdf
【redis源码学习】持久化机制,java程序员面试算法宝典pdf
|
2天前
|
算法 架构师 网络协议
对标腾讯T9架构师的 Android 面试题新鲜出炉,算法真的太重要了
对标腾讯T9架构师的 Android 面试题新鲜出炉,算法真的太重要了
|
2天前
|
移动开发 算法 搜索推荐
2024最新Android算法相关面试大全,请查收
2024最新Android算法相关面试大全,请查收
|
3天前
|
存储 缓存 算法
面试遇到算法题:实现LRU缓存
V哥的这个实现的关键在于维护一个双向链表,它可以帮助我们快速地访问、更新和删除最近最少使用的节点,同时使用哈希表来提供快速的查找能力。这样,我们就可以在 O(1) 的时间复杂度内完成所有的缓存操作。哈哈干净利索,回答完毕。
|
3天前
|
算法 搜索推荐 Python
数据结构与算法在Python面试中的应用实例
【4月更文挑战第13天】本文聚焦Python面试中的数据结构与算法问题,包括排序算法、链表操作和树图遍历。重点讨论了快速排序、链表反转和二叉树前序遍历的实现,并指出理解算法原理、处理边界条件及递归操作是避免错误的关键。通过实例代码和技巧分享,帮助面试者提升面试表现。
13 0