秋招算法备战第31天 | 贪心算法理论基础、455.分发饼干、376. 摆动序列、53. 最大子序和

chatgpt/2023/9/24 1:19:19

贪心算法理论基础

  • 贪心算法并没有固定的套路,唯一的难点就是如何通过局部最优,推出整体最优。
  • 如何验证可不可以用贪心算法呢?最好用的策略就是举反例,如果想不到反例,那么就试一试贪心吧。
  • 刷题或者面试的时候,手动模拟一下感觉可以局部最优推出整体最优,而且想不到反例,那么就试一试贪心。

贪心算法一般分为如下四步:

  • 将问题分解为若干个子问题
  • 找出适合的贪心策略
  • 求解每一个子问题的最优解
  • 将局部最优解堆叠成全局最优解

这个四步其实过于理论化了,我们平时在做贪心类的题目 很难去按照这四步去思考,真是有点“鸡肋”。做题的时候,只要想清楚 局部最优 是什么,如果推导出全局最优,其实就够了。

455. 分发饼干 - 力扣(LeetCode)

这是一个经典的贪心算法问题,策略是优先满足胃口小的孩子。具体操作就是先将孩子的胃口数组和饼干尺寸数组排序,然后从小到大匹配。如果当前饼干可以满足当前孩子,就把饼干给孩子,然后移动到下一个孩子和下一个饼干;如果当前饼干无法满足当前孩子,就放弃当前饼干,移动到下一个饼干。

这种算法的理论支持是,如果当前饼干无法满足当前孩子,那么它也无法满足胃口更大的孩子,因此应该放弃。如果可以满足当前孩子,就应该先满足他,因为下一个饼干不一定能满足下一个孩子,但是下一个饼干可能会满足当前孩子或者胃口更大的孩子。

Python实现代码如下:

def findContentChildren(g, s):g.sort()s.sort()child_i = cookie_j = 0while child_i < len(g) and cookie_j < len(s):if s[cookie_j] >= g[child_i]:child_i += 1cookie_j += 1return child_i

在这段代码中,g代表孩子的胃口数组,s代表饼干的尺寸数组。函数返回的是能够满足的孩子数量。数组gs都先进行了排序,然后用两个指针child_icookie_j分别遍历孩子和饼干。如果当前饼干能满足当前孩子,就将饼干给孩子,然后考虑下一个孩子和下一个饼干;否则放弃当前饼干,考虑下一个饼干。这样一直进行,直到没有孩子或者饼干为止。最后返回满足的孩子数量child_i,即为结果。

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

贪心算法

在这个问题中,贪心算法的策略是我们始终尽可能地使序列保持摆动。换句话说,我们总是优先考虑改变趋势,即如果当前是上升的,我们就寻找下一个下降的点,反之亦然。

这个问题实际上是找出数组中的所有"转折点"。一个"转折点"是指该点两侧的差值与该点和其前一点的差值异号。也就是说,如果该点比前一点大,那么它应该比后一点小;反之亦然。

这种策略可以通过一次遍历完成,时间复杂度是O(n)。

Python实现代码如下:

def wiggleMaxLength(nums):n = len(nums)if n < 2:return nprevdiff = nums[1] - nums[0]count = 2 if prevdiff != 0 else 1for i in range(2, n):diff = nums[i] - nums[i - 1]if (diff > 0 and prevdiff <= 0) or (diff < 0 and prevdiff >= 0):count += 1prevdiff = diffreturn count

在这段代码中,nums代表给定的整数数组。函数返回的是最长摆动子序列的长度。首先计算前两个数字的差值prevdiff,然后从第三个数字开始,如果当前数字与前一个数字的差值diffprevdiff异号,说明当前数字是一个转折点,摆动序列长度增加1,然后更新prevdiff为当前的差值。最后返回摆动序列长度count,即为结果。

这个算法的时间复杂度是O(n),空间复杂度是O(1),满足题目的要求。

动态规划

这是一道典型的动态规划问题,我们需要找到数组中的最长摆动子序列的长度。

我们可以维护两个动态规划数组 updown,其中 up[i] 表示在到达数组的第 i 个位置时,最后一次摆动是上升的最长子序列长度,down[i] 表示在到达数组的第 i 个位置时,最后一次摆动是下降的最长子序列长度。

状态转移方程为:

  • 如果 nums[i] > nums[i - 1],那么我们可以在以 i - 1 结尾的下降子序列后面接一个 nums[i],形成一个更长的摆动序列,所以有 up[i] = down[i - 1] + 1down[i] = down[i - 1]
  • 如果 nums[i] < nums[i - 1],那么我们可以在以 i - 1 结尾的上升子序列后面接一个 nums[i],形成一个更长的摆动序列,所以有 down[i] = up[i - 1] + 1up[i] = up[i - 1]
  • 如果 nums[i] == nums[i - 1],那么我们无法在子序列后面接一个 nums[i] 形成一个更长的摆动序列,所以有 up[i] = up[i - 1]down[i] = down[i - 1]

最后的答案就是 up[n - 1]down[n - 1] 的最大值。

Python 代码如下:

def wiggleMaxLength(nums):n = len(nums)if n < 2:return nup = [0] * ndown = [0] * nup[0] = down[0] = 1for i in range(1, n):if nums[i] > nums[i - 1]:up[i] = max(up[i - 1], down[i - 1] + 1)down[i] = down[i - 1]elif nums[i] < nums[i - 1]:up[i] = up[i - 1]down[i] = max(up[i - 1] + 1, down[i - 1])else:up[i] = up[i - 1]down[i] = down[i - 1]return max(up[-1], down[-1])

这段代码中,nums 代表给定的整数数组。函数返回的是最长摆动子序列的长度。数组 updown 存储了以每个位置结尾的最长上升和下降摆动子序列的长度。然后通过状态转移方程更新 updown,最后返回 updown 的最大值,即为最长摆动子序列的长度。

这个算法的时间复杂度是 O(n),空间复杂度是 O(n),满足题目的要求。

53. 最大子数组和 - 力扣(LeetCode)

贪心算法

贪心算法的思路是:遍历数组,并在每个步骤中维护当前的子数组和以及到目前为止找到的最大子数组和。如果当前子数组和变为负数,则在下一个元素处开始新的子数组。

Python实现代码如下:

def maxSubArray(nums):current_sum = max_sum = nums[0]for num in nums[1:]:current_sum = max(num, current_sum + num)max_sum = max(max_sum, current_sum)return max_sum

动态规划

动态规划的思路稍微复杂一些。设dp[i]表示以第i个元素结尾的最大子数组和,那么dp[i]可以由dp[i-1]和nums[i]决定。如果dp[i-1]大于0,dp[i]就等于dp[i-1]加上nums[i],否则等于nums[i]。我们要求的结果就是dp数组中的最大值。

Python实现代码如下:

def maxSubArray(nums):dp = [0] * len(nums)dp[0] = nums[0]for i in range(1, len(nums)):dp[i] = max(dp[i-1] + nums[i], nums[i])return max(dp)

两种算法的时间复杂度都是O(n),空间复杂度上,贪心算法是O(1),动态规划是O(n),如果将动态规划的解法优化一下,可以降低到O(1),因为dp[i]只依赖于dp[i-1]。

优化后的动态规划解法:

def maxSubArray(nums):max_sum = curr_sum = nums[0]for num in nums[1:]:curr_sum = max(curr_sum + num, num)max_sum = max(max_sum, curr_sum)return max_sum

这个解法与贪心算法看起来非常相似,但它们的思考方式是不同的。贪心算法是在每一步都采取局部最优解,而动态规划则是在每一步都根据前一步的结果做出决策。

总结

贪心算法是在每一步都采取局部最优解,而动态规划则是在每一步都根据前一步的结果做出决策。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.exyb.cn/news/show-5313549.html

如若内容造成侵权/违法违规/事实不符,请联系郑州代理记账网进行投诉反馈,一经查实,立即删除!

相关文章

docker简单web管理docker.io/uifd/ui-for-docker

要先pull这个镜像docker.io/uifd/ui-for-docker 这个软件默认只能使用9000端口&#xff0c;别的不行&#xff0c;因为作者在镜像制作时已加入这一层 刚下下来镜像可以通过docker history docker.io/uifd/ui-for-docker 查看到这个端口已被 设置 如果在没有设置br0网关时&…

英语语法 基础知识(5大基本句型,9大句子成分,10大词性)

目录 一、英语五大基本句型 1.主语 谓语(不及物动词) (状语) : 1 介绍 2 示例 2.主语 谓语(及物动词) 宾语 : 1 介绍 2 示例 3.主语 谓语 间接宾语 直接宾语 : 1 介绍 2 示例 4.主语 谓语 宾语 宾语补足语 : 1 介绍 2 示例 5.主语 系动词 表语 : 1 介绍 2 …

Unity源码分享-黄金矿工游戏完整版

Unity源码分享-黄金矿工游戏完整版 项目地址&#xff1a;https://download.csdn.net/download/Highning0007/88118933

Hudi Flink SQL源码调试学习(1)

前言 本着学习hudi-flink源码的目的&#xff0c;利用之前总结的文章Hudi Flink SQL代码示例及本地调试中的代码进行调试,记录调试学习过程中主要的步骤及对应源码片段。 版本 Flink 1.15.4Hudi 0.13.0 目标 在文章Hudi Flink SQL代码示例及本地调试中提到&#xff1a;我们…

力扣75——深度优先搜索

总结leetcode75中深度优先搜索的算法题解题思路。 上一篇&#xff1a;力扣75——链表 以下代码部分为本人所写&#xff0c;部分为官方示例代码。 力扣75——深度优先搜索 1 二叉树的最大深度2 叶子相似的树3 统计二叉树中好节点的数目4 路径总和 III5 二叉树中的最长交错路径6 …

【C/C++】类之间的纵向关系——继承的概念

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; &#x1f525;c系列专栏&#xff1a;C/C零基础到精通 &#x1f525; 给大…

vue-cli4升级到vue-cli5的踩坑记录

前言 最近对部分项目升级了vue-cli脚手架&#xff0c;记录一下 问题一&#xff1a; scss/less/css中无法引入public下的静态资源 问题描述 在样式文件中使用静态资源路径导致编译无法通过 错误信息如下&#xff1a; Module not found: Error: Cant resolve /img/login/lo…

无涯教程-jQuery - scrollLeft( val )方法函数

scrollLeft(val)方法用于将所有匹配元素上的向左滚动偏移量设置为传递的值。 此方法适用于可见和隐藏元素。 scrollLeft( val ) - 语法 selector.scrollLeft( val ) 这是此方法使用的所有参数的描述- val - 代表所需滚动左偏移量的正数。 scrollLeft( val ) - 示例 以…
推荐文章