【LeetCode 刷题】贪心算法(2)-进阶

此博客为《代码随想录》二叉树章节的学习笔记,主要内容为贪心算法进阶的相关题目解析。

文章目录

  • [135. 分发糖果](#135. 分发糖果)
  • [406. 根据身高重建队列](#406. 根据身高重建队列)
  • [134. 加油站](#134. 加油站)
  • [968. 监控二叉树](#968. 监控二叉树)

135. 分发糖果

题目链接

python 复制代码
class Solution:
    def candy(self, ratings: List[int]) -> int:
        n = len(ratings)
        left = [1] * n
        for i in range(1, n):
            if ratings[i] > ratings[i-1]:
                left[i] = left[i-1] + 1
        right = [1] * n
        res = left[n-1]
        for i in range(n - 2, -1, -1):
            if ratings[i] > ratings[i+1]:
                right[i] = right[i+1] + 1
            res += max(left[i], right[i])
        return res
  • 将同时考虑两侧的元素分解为左右两个规则:
    • 左规则:只考虑当前元素和其左侧的元素,使其满足题目要求
    • 右规则:只考虑当前元素和其右侧的元素,使其满足题目要求
  • 先从左至右遍历,获得满足左规则的序列;再从右至左遍历,获得满足右规则的序列;最后每个位置取两序列的最大值,累加即为答案

406. 根据身高重建队列

题目链接

python 复制代码
class Solution:
    def reconstructQueue(self, people: List[List[int]]) -> List[List[int]]:
        people.sort(key=lambda x: (-x[0], x[1]))
        res = []
        for p in people:
            if p[1] < len(res):
                res.insert(p[1], p)
            else:
                res.append(p)
        return res
  • 先按照身高降序排序,然后按照排位升序排序 ,按照排序后的数组顺序处理,根据当前元素的排位在结果序列中动态插队(因为按身高降序排列后,比当前元素高的元素都已经被处理过且放入 res 中,而后续未处理的元素又比当前元素矮,不影响结果)

134. 加油站

题目链接

python 复制代码
class Solution:
    def canCompleteCircuit(self, gas: List[int], cost: List[int]) -> int:
        cur_gas, sum_gas = 0, 0
        res = 0
        for i in range(len(gas)):
            cur_gas += gas[i] - cost[i]
            sum_gas += gas[i] - cost[i]
            if cur_gas < 0:
                res = i + 1
                cur_gas = 0
        return res if sum_gas >= 0 else -1
  • 从 0 号站点开始累加,一旦 cur_sum 小于零,则表示之前所有的站点都不可能作为起点,选择下一个位置作为新的起点,cur_sum 重新开始累加
  • sum_gas 在遍历过程中累计全程所有的油量,如果 sum_gas < 0 则表示不可能开完一圈,返回 -1。

968. 监控二叉树

题目链接

python 复制代码
class Solution:
    def minCameraCover(self, root: Optional[TreeNode]) -> int:
        # 0 无覆盖 1 有覆盖 2 有摄像头
        def traversal(node: Optional[TreeNode]) -> int:
            if not node:
                return 1
            left = traversal(node.left)
            right = traversal(node.right)
            if left == 1 and right == 1:
                return 0
            if left == 0 or right == 0:
                nonlocal res
                res += 1
                return 2
            if left == 2 or right == 2:
                return 1
        res = 0
        val = traversal(root)
        if val == 0: res += 1  # 根结点无覆盖
        return res
  • 贪心准则:尽可能不要让覆盖的范围重合
  • 设置三个节点的状态:0 无覆盖;1 有覆盖;2 有摄像头
    • 两个子节点有任何一个为无覆盖的状态,则当前节点需要放置摄像头
    • 两个子节点有任何一个有摄像头,则当前节点为有覆盖状态
    • 两个子节点都有覆盖(说明是孙子节点的摄像头),则当前节点为无覆盖
  • 根据贪心准则,叶子结点是不需要放置摄像头的,因此将空节点返回的状态值设置为 1 有覆盖,这样叶子结点的状态值则为 0 无覆盖,满足算法要求
  • 最后判断根节点的状态,如果为无覆盖,则需要额外再放一个摄像头
相关推荐
董董灿是个攻城狮2 小时前
5分钟搞懂什么是窗口注意力?
算法
Dann Hiroaki2 小时前
笔记分享: 哈尔滨工业大学CS31002编译原理——02. 语法分析
笔记·算法
蹦蹦跳跳真可爱5893 小时前
Python----OpenCV(图像増强——高通滤波(索贝尔算子、沙尔算子、拉普拉斯算子),图像浮雕与特效处理)
人工智能·python·opencv·计算机视觉
nananaij4 小时前
【Python进阶篇 面向对象程序设计(3) 继承】
开发语言·python·神经网络·pycharm
雷羿 LexChien4 小时前
从 Prompt 管理到人格稳定:探索 Cursor AI 编辑器如何赋能 Prompt 工程与人格风格设计(上)
人工智能·python·llm·编辑器·prompt
qqxhb4 小时前
零基础数据结构与算法——第四章:基础算法-排序(上)
java·数据结构·算法·冒泡·插入·选择
敲键盘的小夜猫4 小时前
LLM复杂记忆存储-多会话隔离案例实战
人工智能·python·langchain
高压锅_12205 小时前
Django Channels WebSocket实时通信实战:从聊天功能到消息推送
python·websocket·django
FirstFrost --sy5 小时前
数据结构之二叉树
c语言·数据结构·c++·算法·链表·深度优先·广度优先
森焱森6 小时前
垂起固定翼无人机介绍
c语言·单片机·算法·架构·无人机