当前位置: 首页 > 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

相关文章:

  • 2262 1527的解码实验程序
  • 弥散制氧机工作机制:高原低氧环境的氧浓度重构技术
  • FPGA实现CNN卷积层:高效窗口生成模块设计与验证
  • 在 Ubuntu 上挂载其他硬盘的步骤
  • docker-记录一次容器日志<container_id>-json.log超大问题的处理
  • Redis--缓存穿透与缓存雪崩详解及解决方案
  • 基于MATLAB的二维圆形随机骨料生成程序
  • 深入解析 CountDownLatch、Semaphore 和CyclicBarrier
  • 微服务FallbackFactory和FallbackClass
  • ROS2学习(16)------ URDF 机器人建模方法
  • ChatGPT + 知网 + 知乎,如何高效整合信息写出一篇专业内容?
  • 逛网站看到个场景题,如何防止重复下单
  • 一招解决 win10 安装 Abobe PR/AE 打不开或闪退
  • 股票收益率的计算
  • 【R语言编程绘图-调色】
  • 智能嗅探AJAX触发:机器学习在动态渲染中的创新应用
  • Mini-F5265-OB开发板——UART不定长接收
  • 《自动驾驶轨迹规划实战:Lattice Planner实现避障路径生成(附可运行Python代码)》—— 零基础实现基于离散优化的避障路径规划
  • 数据库管理-第330期 数据库国产化可以顺便做的事情(20250528)
  • git stash 的使用
  • 我国网站开发/网站推广具体内容
  • wordpress 4.8.1/seo网站有优化培训吗
  • 网络设计方案书主要包括哪些内容/优化网站seo策略
  • 大丰做网站哪家公司好/黄石seo诊断
  • 公司哪家好点/中山seo关键词
  • 淘宝上做网站的客服聊天技巧/怎么做网络营销推广