当前位置: 首页 > news >正文

贪心算法实战篇2

文章目录

  • 前言
  • 序列问题
    • 摆动序列
    • 单调递增的数字
  • 贪心解决股票问题
    • 买卖股票的最佳时机II
  • 两个维度权衡问题
    • 分发糖果
    • 根据身高重建队列

前言

今天继续带大家进行贪心算法的实战篇2,本章注意来解答一些运用贪心算法的中等的问题,大家好好体会,怎么从构建局部最优到全局最优的。一文带大家弄懂。本文用于记录自己的学习过程,同时向大家进行分享相关的内容。本文内容参考于代码随想录同时包含了自己的许多学习思考过程,如果有错误的地方欢迎批评指正!

image-20250516165753896

序列问题

摆动序列

376. 摆动序列 - 力扣(LeetCode)

image-20250516170811819

**相关技巧:**首先我们先来理解题意要我们干什么,就是在一个序列中,找出其摆动序列,即其差值的形式为正负交替的,找出其最长子序列的长度。就是我们可以通过删除一些元素来使其满足摆动序列的性质,这里就是要求其最大的长度。

理解了题意再来看我们该怎么做这道题。我们通过什么样的形式来删除一些不必要的元素呢?举个例子来说,当序列为1,2,3,4时,可以很明显的看出最大长度为2,但是是哪个组合呢?1,2可以,1,3可以,1,4也可以。这里我们注意,我们以最后的作为答案。即我们一直增的话,我们就找到山顶,一直降的话我们就找到山谷,像爬山一样,找出山顶山谷的过程,这就是我们使用的贪心策略。

这里代码实现的关键就是我们怎么找到山谷山顶,其实很简单,我们所需要的摆动序列差值是正负摆动的,即两者差小于零,但是差值为零就不算摆动了,所以这样我们就能够很容易的写出代码了,如下所示。

class Solution:def wiggleMaxLength(self, nums: List[int]) -> int:if len(nums) <= 1:return len(nums)  # 如果数组长度为0或1,则返回数组长度preDiff,curDiff ,result  = 0,0,1  #题目里nums长度大于等于1,当长度为1时,其实到不了for循环里去,所以不用考虑nums长度for i in range(len(nums) - 1):curDiff = nums[i + 1] - nums[i]if curDiff * preDiff <= 0 and curDiff !=0:  #差值为0时,不算摆动result += 1preDiff = curDiff  #如果当前差值和上一个差值为一正一负时,才需要用当前差值替代上一个差值return result

单调递增的数字

738. 单调递增的数字 - 力扣(LeetCode)

image-20250516170836651

**相关技巧:**这题题目还是很好理解了,就是找到单调递增的最大数,首先我们来看看,我们该怎么去遍历这个数,如果我们从左往右进行遍历,当前数小于上一位了,修改上一位,可是修改之后呢?上一位因为修改又小于上上一位了怎么办?那我们在来看从右往左进行遍历,当前位比前一小,说明需要修改前一位,修改之后,我们再将修改位之后的都变成9,这样我们就不会破坏递增性质了。

比如说就是332,第2位的3比2大,不符合递增性质,所以,3-1变成2,修位之后都变成9,就变成了329,然后再看第1位的3比第2位的2大,所以再修改,3-1变成2,第一位的3之后的都变成9,就变成了299,一次遍历就能够得到我们需要的,代码如下。

class Solution:def monotoneIncreasingDigits(self, n: int) -> int:# 将整数转换为字符串strNum = list(str(n))# 从右往左遍历字符串for i in range(len(strNum) - 1, 0, -1):# 如果当前字符比前一个字符小,说明需要修改前一个字符if strNum[i - 1] > strNum[i]:strNum[i - 1] = str(int(strNum[i - 1]) - 1)  # 将前一个字符减1# 将修改位置后面的字符都设置为9,因为修改前一个字符可能破坏了递增性质strNum[i:] = '9' * (len(strNum) - i)# 将列表转换为字符串,并将字符串转换为整数并返回return int(''.join(strNum))

贪心解决股票问题

买卖股票的最佳时机II

122. 买卖股票的最佳时机 II - 力扣(LeetCode)

image-20250516170859647

**相关技巧:**首先我们要清楚,我们一天只能持有一支股票,并且我们一天中只有买股票和卖股票的操作。那么我们需要如何才能获得最大的利润呢?

首先一个非常重要的就是要知道,利润是可以分解的。

就是说我们假如第 0 天买入,第 3 天卖出,那么利润为:prices[3] - prices[0]。其本质相当于(prices[3] - prices[2]) + (prices[2] - prices[1]) + (prices[1] - prices[0])。

**此时就是把利润分解为每天为单位的维度,而不是从 0 天到第 3 天整体去考虑!**那么根据 prices 可以得到每天的利润序列:(prices[i] - prices[i - 1])…(prices[1] - prices[0])。

这个时候再来看,我们需要获得最大的利润是不是就只需要将每天正的利润加起来即可。其实贪心也体现再其中,我只要赚了就卖掉,就是很典型的贪心思想了。

class Solution:def maxProfit(self, prices: List[int]) -> int:result = 0for i in range(1, len(prices)):result += max(prices[i] - prices[i - 1], 0)return result

两个维度权衡问题

分发糖果

135. 分发糖果 - 力扣(LeetCode)

image-20250516170945614

**相关技巧:**首先看题,我们的第一想法肯定就是一次遍历,如果其分数比左右两边大,就取大的加1,但是这样我们同时兼顾就会造成顾此失彼的状态,比如说如下的情况

image-20250528163442809

一开始都是1个,即是1,1,1,1,1,1,1。按照我们刚才所说的同时比较两边就会得到最终的1,1,1,2,2,2,1。但是这最终得到的是不符合题意的,最高分的5应该比边上的大,但最终得到的却不是的。

所以我们取做的时候就要分维度比较讨论。首先从左往右遍历比较,并且比较的是其右边的数,然后从右往左遍历比较,并且比较的是其左边的数,最终计算糖果数量。代码如下,还是比较容易理解的。

class Solution:def candy(self, ratings: List[int]) -> int:n = len(ratings)candies = [1] * n# Forward pass: handle cases where right rating is higher than leftfor i in range(1, n):if ratings[i] > ratings[i - 1]:candies[i] = candies[i - 1] + 1# Backward pass: handle cases where left rating is higher than rightfor i in range(n - 2, -1, -1):if ratings[i] > ratings[i + 1]:candies[i] = max(candies[i], candies[i + 1] + 1)return sum(candies)

根据身高重建队列

406. 根据身高重建队列 - 力扣(LeetCode)

image-20250516171010146

**相关技巧:**同样的这回我们仍旧是需要从两个维度来进行考量。首先考虑身高,然后再考虑k,我们首先按照身高排序,然后执行插入操作,按身高先插入,再按k值进行插入,当k值相同的时候,身高矮的在前面。其实就是两次排序的过程,先按身高排,再按k值排。其实其贪心策略就如下:

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

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

实现的代码如下,这还是比较好理解的。

class Solution:def reconstructQueue(self, people: List[List[int]]) -> List[List[int]]:# 先按照h维度的身高顺序从高到低排序。确定第一个维度# lambda返回的是一个元组:当-x[0](维度h)相同时,再根据x[1](维度k)从小到大排序people.sort(key=lambda x: (-x[0], x[1]))que = []# 根据每个元素的第二个维度k,贪心算法,进行插入# people已经排序过了:同一高度时k值小的排前面。for p in people:que.insert(p[1], p)return que
http://www.xdnf.cn/news/702127.html

相关文章:

  • mimics导出图像 标注文件
  • 学习日记-day18-5.28
  • 央国企迁移国产数据库:数据迁移5步法与4项管理准则
  • GATED DELTA NETWORKS : IMPROVING MAMBA 2 WITH DELTA RULE
  • 【AI算法工程师面试指北】小球检测问题
  • 【Python-Day 19】函数的回响:深入理解 `return` 语句与返回值
  • 融智学视域下的多时空统一框架与信智序位法则
  • 基于CATIA参数化圆锥建模的自动化插件开发实践——NX建模之圆锥体命令的参考与移植(三)
  • 图神经网络在信息检索重排序中的应用:原理、架构与Python代码解析
  • ORB-SLAM2学习笔记:ORBextractor::operator()函数的逐行解析
  • 应用宝的NotificationManagerService_post_com.tencent.android.qqdownloader持锁现象
  • 涨薪技术|0到1学会性能测试第87课-Webservice接口性能测试
  • (nice!!!)(LeetCode 每日一题) 3372. 连接两棵树后最大目标节点数目 I (贪心+深度优先搜索dfs)
  • GPU时间与transformer架构计算量分析
  • qemu安装risc-V 64
  • springboot配置mybatis debug的sql日志输出
  • DelayQueue源码解析
  • 《活法》
  • Python实例题:Python实现FTP弱口令扫描器
  • 如何去除文章的AI痕迹2025新方法
  • DeepSeek 工作应用深度指南
  • 二叉树的锯齿形层序遍历——灵活跳跃的层次结构解析
  • 第十一节:第三部分:异常:异常的两种处理方式
  • 【Unity】自动生成围绕模型的路径点
  • 企业应如何构建用户画像系统
  • C语言Day9:C语言类型转换规则
  • Linux Crash工具全解:内核崩溃分析的一切
  • shell脚本总结11
  • 华为OD机试真题——矩形绘制(2025A卷:200分)Java/python/JavaScript/C/C++/GO最佳实现
  • 数据库表与实体类设计