代码随想录学习Day 30

860.柠檬水找零

题目链接

讲解链接

思路:需要找零的情况是顾客支付10元或20元,尤其是支付20元时需要考虑找零的方式,此时可以选择找零3张5元或者一张10元+一张5元,按照贪心算法的思路来看:

局部最优:在找零20元时尽量采用10+5的方式,多保留5元,因为5元能够找零的情况更多;

全局最优:每次找零尽可能多的保留5元从而为更多的顾客找零。

找到贪心的思路后其余部分实现就比较简单了。用一个字典来统计当前手中的钞票,每次找零时将消耗的钞票-1,如果当前钞票不满足找零条件则返回False,循环结束返回True。

python 复制代码
class Solution:
    def lemonadeChange(self, bills: List[int]) -> bool:
        if bills[0] > 5:  # 第一个就大于5无法找零,直接返回
            return False
        money = {5:0, 10:0, 20:0}  # 初始手中没钱
        for i in range(len(bills)):  # 遍历账单数组
            money[bills[i]] = money.get(bills[i], 0) + 1  # 每次将收到的钱在字典中对应位置+1
            if bills[i] == 5:  # 等于5则不需要找零
                continue
            elif bills[i] == 10:  # 等于10需要用一张5元来找零
                if money[5] > 0:  # 当前手中有5元
                    money[5] -= 1  # 5元钞票的数量-1
                    continue
                else:
                    return False
            elif bills[i] == 20:  # 需要用3张5元或者1张10元和一张5元来找零
                if money[5] >= 1 and money[10] >= 1:  # 优先考虑后者,因为要尽量多保留5元
                    money[5] -= 1
                    money[10] -= 1
                elif money[5] >= 3:  # 在没有10+5的情况下才考虑3*5
                    money[5] -= 3
                    continue
                else:
                    return False
        return True  # 循环结束则返回True

406.根据身高重建队列

题目链接

讲解链接

思路:先对所有人按照身高从大到小进行排序,身高相同的话则k小的站前面,接下来以k为下标插入队列即可。因为按照身高排序之后,优先按身高高的people的k来插入,后序插入节点也不会影响前面已经插入的节点,最终按照k的规则完成了队列。

局部最优:优先按身高高的people的k来插入。插入操作过后的people满足队列属性

全局最优:最后都做完插入操作,整个队列满足题目队列属性

python 复制代码
class Solution:
    def reconstructQueue(self, people: List[List[int]]) -> List[List[int]]:
        people.sort(key=lambda x: (-x[0], x[1]))  # 先按照身高降序排序,在按照k值从小到大排序
        queue = []
        for p in people:  # 按照k值进行插入
            queue.insert(p[1], p)  # people已经排序过了,同一高度时k值小的排前面
        return queue

452.用最少数量的箭引爆气球

题目链接

讲解链接

局部最优:当气球出现重叠,一起射,所用弓箭最少。

全局最优:把所有气球射爆所用弓箭最少。

为了让气球尽可能的重叠,需要对数组进行排序。按照起始位置排序,那么就从前向后遍历气球数组,靠左尽可能让气球重复。从前向后遍历遇到重叠的气球,重叠气球中右边边界的最小值之前的区间一定需要一支箭。可以看出首先第一组重叠气球,一定需要一支箭。而气球3的左边界大于第一组重叠气球的最小右边界,所以再需要一支箭来射气球3。

python 复制代码
class Solution:
    def findMinArrowShots(self, points: List[List[int]]) -> int:
        points.sort(key=lambda x: x[0])  # 将所有气球按照左端点排序
        left, right = points[0][0],points[0][1]  # 初始化左右边界值为第一个气球的左右端点
        count = 1  # 统计需要箭的数量,初始为1
        for i in points:  # 遍历气球数组
            if i[0] > right:  # 如果当前气球的左端点大于右边界值,说明不能只用一支箭将其与前面的气球引爆,消耗箭矢的数量+1
                count += 1
                left, right = i[0], i[1]  # 更新当前的左右边界,为该气球的左右端点
            else:  # 如果不符合上面的条件,则说明可以用一支箭将该气球与之前的气球引爆
                left = max(left, i[0])  # 更新左边界,取区间的交集
                right = min(right, i[1])  # 同上取交集更新右边界
        return count
相关推荐
小唐C++5 分钟前
C++小病毒-1.0勒索
开发语言·c++·vscode·python·算法·c#·编辑器
文城52120 分钟前
Mysql存储过程(学习自用)
数据库·学习·mysql
北 染 星 辰33 分钟前
Python网络自动化运维---用户交互模块
开发语言·python·自动化
codists37 分钟前
《CPython Internals》阅读笔记:p336-p352
python
我们的五年1 小时前
【C语言学习】:C语言补充:转义字符,<<,>>操作符,IDE
c语言·开发语言·后端·学习
Мартин.1 小时前
[Meachines] [Easy] GoodGames SQLI+Flask SSTI+Docker逃逸权限提升
python·docker·flask
日日行不惧千万里1 小时前
如何用YOLOv8训练一个识别安全帽的模型?
python·yolo
Icoolkj1 小时前
微服务学习-Nacos 注册中心实战
linux·学习·微服务
siy23331 小时前
【c语言日寄】Vs调试——新手向
c语言·开发语言·学习·算法
无涯学徒19982 小时前
R6学习打卡
学习