算法学习——LeetCode力扣贪心篇1

算法学习——LeetCode力扣贪心篇1

在这里插入图片描述

455. 分发饼干

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

描述

假设你是一位很棒的家长,想要给你的孩子们一些小饼干。但是,每个孩子最多只能给一块饼干。

对每个孩子 i,都有一个胃口值 g[i],这是能让孩子们满足胃口的饼干的最小尺寸;并且每块饼干 j,都有一个尺寸 s[j] 。如果 s[j] >= g[i],我们可以将这个饼干 j 分配给孩子 i ,这个孩子会得到满足。你的目标是尽可能满足越多数量的孩子,并输出这个最大数值。

示例

示例 1:

输入: g = [1,2,3], s = [1,1]
输出: 1
解释:
你有三个孩子和两块小饼干,3个孩子的胃口值分别是:1,2,3。
虽然你有两块小饼干,由于他们的尺寸都是1,你只能让胃口值是1的孩子满足。
所以你应该输出1。

示例 2:

输入: g = [1,2], s = [1,2,3]
输出: 2
解释:
你有两个孩子和三块小饼干,2个孩子的胃口值分别是1,2。
你拥有的饼干数量和尺寸都足以让所有孩子满足。
所以你应该输出2.

提示

  • 1 <= g.length <= 3 * 104
  • 0 <= s.length <= 3 * 104
  • 1 <= g[i], s[j] <= 231 - 1

代码解析

class Solution {
public:int findContentChildren(vector<int>& g, vector<int>& s) {int num = 0 , s_point = s.size()-1;//饼干和孩子排序sort(g.begin() , g.end());sort(s.begin() , s.end());//先用最大的饼干开始喂能吃饱的孩子,吃不饱的跳过不喂for(int i=g.size()-1 ; i >= 0 && s_point >=0 ;i--){if( s[s_point] >= g[i]){s_point--; //喂好一个饼干指针减少num++;}}return num;}
};

376. 摆动序列

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

描述

如果连续数字之间的差严格地在正数和负数之间交替,则数字序列称为 摆动序列 。第一个差(如果存在的话)可能是正数或负数。仅有一个元素或者含两个不等元素的序列也视作摆动序列。

例如, [1, 7, 4, 9, 2, 5] 是一个 摆动序列 ,因为差值 (6, -3, 5, -7, 3) 是正负交替出现的。

相反,[1, 4, 7, 2, 5] 和 [1, 7, 4, 5, 5] 不是摆动序列,第一个序列是因为它的前两个差值都是正数,第二个序列是因为它的最后一个差值为零。
子序列 可以通过从原始序列中删除一些(也可以不删除)元素来获得,剩下的元素保持其原始顺序。

给你一个整数数组 nums ,返回 nums 中作为 摆动序列 的 最长子序列的长度 。

示例

示例 1:

输入:nums = [1,7,4,9,2,5]
输出:6
解释:整个序列均为摆动序列,各元素之间的差值为 (6, -3, 5, -7, 3) 。

示例 2:

输入:nums = [1,17,5,10,13,15,10,5,16,8]
输出:7
解释:这个序列包含几个长度为 7 摆动序列。
其中一个是 [1, 17, 10, 13, 10, 16, 8] ,各元素之间的差值为 (16, -7, 3, -3, 6, -8) 。

示例 3:

输入:nums = [1,2,3,4,5,6,7,8,9]
输出:2

提示

  • 1 <= nums.length <= 1000
  • 0 <= nums[i] <= 1000

进阶

你能否用 O(n) 时间复杂度完成此题?

代码解析

贪心法

核心思想是找山坡。接替找上山坡和下山坡
在开始由前两个点确定向上还是向下
开始向上,就找向下
开始向下就找向上

在这里插入图片描述

class Solution {
public:int wiggleMaxLength(vector<int>& nums) {if(nums.size()==1 ||(nums.size()==2 && nums[0] != nums[1]) ) return nums.size();int pre = 0 ,num = 0 ; bool flag ,start = false ;for(int i=1 ,pre = 0; i<nums.size() ;i++ , pre++) //遍历所有点,pre是前一个点,i是当前点{if(nums[i] == nums[pre]) continue; //如果遇到相同点跳过if(start == false)//start是开始标志,当第一次开始时,确定第一个坡是上山还是下山{if (nums[i] > nums[pre] ) flag = 0; //第一个坡是上山,flag为0,找下山坡else flag = 1;                      //第一个坡是下山,flag为1,找上山坡start = true;  //开始正式循环找num++;//第一个坡记录}else if(start == true){if(   (flag == 1 && (nums[i] > nums[pre]))      //上一个是下坡,找到上坡了||(flag == 0 && (nums[i] < nums[pre]))    ) //上一个是上坡,找到下坡了{num++; //记录坡数flag = !flag;//标志反转,找完上接着下,进行摆动}  }}//num是坡的数量,第一个坡是俩点,补上第一个点+1return num+1;}
};
dp法
class Solution {
public:int wiggleMaxLength(vector<int>& nums) {if(nums.size()==1) return 1;vector<int> dp(nums.size(),1);int flag = -1;for(int i=1 ; i<nums.size();i++){if(flag==-1){if(nums[i]-nums[i-1] > 0) {flag=1;dp[i] = dp[i-1] +1;}else if(nums[i]-nums[i-1] < 0){flag=0;dp[i] = dp[i-1] +1;} continue;}if(nums[i]-nums[i-1] > 0 && flag==0 ){dp[i] = dp[i-1] + 1;flag = 1;}else if(nums[i]-nums[i-1] < 0 && flag==1){dp[i] = dp[i-1] + 1;flag = 0;}else{dp[i]=dp[i-1];continue;}// cout<<dp[i]<<" ";}return dp[nums.size()-1];}
};

53. 最大子数组和

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

描述

给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

子数组 是数组中的一个连续部分。

示例

示例 1:

输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。

示例 2:

输入:nums = [1]
输出:1

示例 3:

输入:nums = [5,4,-1,7,8]
输出:23

提示

  • 1 <= nums.length <= 105
  • -104 <= nums[i] <= 104

进阶

如果你已经实现复杂度为 O(n) 的解法,尝试使用更为精妙的 分治法 求解。

代码解析

贪心法
class Solution {
public:int maxSubArray(vector<int>& nums) {int sum=0 ,result= INT32_MIN;      //sum是当前数组的和,result是sum中最大的时候for(int i=0 ; i<nums.size() ;i++){sum += nums[i];  //记录当前的sumif(sum > result) result= sum;  //如果sum大于当前result,更新resultif(sum < 0) sum = 0;  //某一个时期的sum小于0舍去}return result;}
};
动态规划
class Solution {
public:int maxSubArray(vector<int>& nums) {vector<int>  dp(nums.size() ,0);int result = INT_MIN;dp[0]= nums[0];for(int i=1 ; i<nums.size() ;i++){dp[i] = max(nums[i],dp[i-1]+nums[i]);}for(int i=0 ; i<nums.size() ;i++) {// cout<<dp[i]<<' ';if(dp[i] > result) result = dp[i];}return result;}
};

122. 买卖股票的最佳时机 II

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

描述

给你一个整数数组 prices ,其中 prices[i] 表示某支股票第 i 天的价格。

在每一天,你可以决定是否购买和/或出售股票。你在任何时候 最多 只能持有 一股 股票。你也可以先购买,然后在 同一天 出售。

返回 你能获得的 最大 利润 。

示例

示例 1:

输入:prices = [7,1,5,3,6,4]
输出:7
解释:在第 2 天(股票价格 = 1)的时候买入,在第 3 天(股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5 - 1 = 4 。
随后,在第 4 天(股票价格 = 3)的时候买入,在第 5 天(股票价格 = 6)的时候卖出, 这笔交易所能获得利润 = 6 - 3 = 3 。
总利润为 4 + 3 = 7 。

示例 2:

输入:prices = [1,2,3,4,5]
输出:4
解释:在第 1 天(股票价格 = 1)的时候买入,在第 5 天 (股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5 - 1 = 4 。
总利润为 4 。

示例 3:

输入:prices = [7,6,4,3,1]
输出:0
解释:在这种情况下, 交易无法获得正利润,所以不参与交易可以获得最大利润,最大利润为 0 。

提示

  • 1 <= prices.length <= 3 * 104
  • 0 <= prices[i] <= 104

代码解析

贪心算法

核心思想是找到利润。然后利润大于0的相加

class Solution {
public:int maxProfit(vector<int>& prices) {vector<int> profit; int result = 0;for(int i=1 ;i < prices.size(); i++){if(prices[i] - prices[i-1] > 0 ) //单日利润大于0的存入profit.push_back(prices[i] - prices[i-1]);}for(int i=0 ; i < profit.size(); i++) //计算利润和result += profit[i]; return result;}
};
动态规划

dp数组的含义:

  • dp[i][0] 表示第i天持有股票所得现金。
  • dp[i][1] 表示第i天不持有股票所得最多现金

如果第i天持有股票即dp[i][0]

  • 第i-1天就持有股票,那么就保持现状,即:dp[i - 1][0]
  • 第i天买入股票,就是昨天不持有股票的所得现金减去今天的股票价格
    即:dp[i - 1][1] - prices[i]

]如果第i天不持有股票即dp[i][1]的情况

  • 第i-1天就不持有股票,那么就保持现状,即:dp[i - 1][1]
  • 第i天卖出股票,所得现金就是按照今天股票佳价格卖出后所得现金
    即:prices[i] + dp[i - 1][0]
class Solution {
public:int maxProfit(vector<int>& prices) {if(prices.size()<=1) return 0;vector<vector<int>> dp(prices.size() , vector<int>( 2,0 ) );dp[0][0] = -prices[0];dp[0][1] = 0;for(int i=1 ; i<prices.size() ;i++){dp[i][0] = max( dp[i-1][0] , dp[i-1][1] - prices[i] );dp[i][1] = max( dp[i-1][1] , dp[i-1][0] + prices[i] );}return  dp[prices.size()-1][1];}
};

55. 跳跃游戏

55. 跳跃游戏 - 力扣(LeetCode)

描述

给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。

判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。

示例

示例 1:

输入:nums = [2,3,1,1,4]
输出:true
解释:可以先跳 1 步,从下标 0 到达下标 1, 然后再从下标 1 跳 3 步到达最后一个下标。

示例 2:

输入:nums = [3,2,1,0,4]
输出:false
解释:无论怎样,总会到达下标为 3 的位置。但该下标的最大跳跃长度是 0 , 所以永远不可能到达最后一个下标。

提示

  • 1 <= nums.length <= 104
  • 0 <= nums[i] <= 105

代码解析

回溯法(超时)
class Solution {
public:bool backtraking(vector<int>& nums  , int target ,int step){if (step == target) return 1; //检验步子能否到达最后点else if (step > target) return 0;if(nums[step]==0) return 0;for(int j=nums[step] ; j >= 1; j--){if(backtraking(nums ,target , step + j )) return 1; }return 0;}bool canJump(vector<int>& nums) {int target = nums.size() - 1;int step = 0;return backtraking(nums,target,step);}
};
贪心
class Solution {
public:bool canJump(vector<int>& nums) {int cover=0;for(int i=0 ; i<nums.size()-1 && i <= cover ; i++) //循环点,保证点是覆盖范围内的{if(i+nums[i] > cover ) cover = i+nums[i]; //如果当前点的范围大于现在范围,范围更新}if(cover >= nums.size()-1) return 1;   //如果范围覆盖最后一个点返回成功else return 0;}
};

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

秒懂百科,C++如此简单丨第十九天:动态规划

目录 动态规划的初步理解 求最短路径数 洛谷 P1002 过河卒 题目描述 输入样例 输出样例 思路 AC Code 动态规划的初步理解 什么是动态规划&#xff1f;最直白的理解就是动态的规划。 那高级一点的理解呢&#xff1f;就是每时每刻都拿着一个小本本&#xff0c;也就是…

【C语言】解析刘谦春晚魔术《守岁共此时》

今年的春晚上刘谦表演了魔术《守岁共此时》&#xff0c;台上台下积极互动&#xff08;尤其是小尼&#xff09;&#xff0c;十分的有趣。刘谦老师的魔术不仅仅是他的高超手法&#xff0c;还有这背后的严谨逻辑&#xff0c;下面我们来用C语言来解析魔术吧。 源代码 #define _CRT…

【数据结构】哈希桶封装出map和set

利用之前的哈希桶封装出unordered_map和unordered_set。 这个封装并不简单&#xff0c;迭代器的使用&#xff0c;模板参数的繁多&#xff0c;需要我们一层一层封装。 map是一个k - v类型&#xff0c;set是k类型&#xff0c;那么就明确了如果需要封装&#xff0c;底层的tables…

【原创 附源码】Flutter集成Apple支付详细流程(附源码)

最近有时间&#xff0c;特意整理了一下之前使用过的Flutter平台的海外支付&#xff0c;附源码及demo可供参考 这篇文章只记录Apple支付的详细流程&#xff0c;其他相关Flutter文章链接如下&#xff1a; 【原创 附源码】Flutter集成谷歌支付详细流程(附源码) 【原创 附源码】F…

【前端高频面试题--虚拟DOM篇】

&#x1f680; 作者 &#xff1a;“码上有前” &#x1f680; 文章简介 &#xff1a;前端高频面试题 &#x1f680; 欢迎小伙伴们 点赞&#x1f44d;、收藏⭐、留言&#x1f4ac;前端高频面试题--虚拟DOM篇 虚拟DOM的理解虚拟DOM的解析过程为什么要用虚拟DOM虚拟DOM与真实DOM的…

Phobos捆绑某数控软件AdobeIPCBroker组件定向勒索

前言 Phobos勒索病毒最早于2019年被首次发现并开始流行起来&#xff0c;该勒索病毒的勒索提示信息特征与CrySiS(Dharma)勒索病毒非常相似&#xff0c;但是两款勒索病毒的代码特征却是完全不一样&#xff0c;近日笔者在逛某开源恶意软件沙箱的时候发现了一款Phobos勒索病毒捆绑…

FreeRTOS 调度算法简述

优先级抢占式调度 本章的示例程序已经演示了 FreeRTOS 在什么时候以及以什么方式选择一个什么 样的任务来执行。  每个任务都赋予了一个优先级。  每个任务都可以存在于一个或多个状态。  在任何时候都只有一个任务可以处于运行状态。  调度器总是在所有处于就…

计算机毕业设计分享-SSM实验室耗材管理系统 13205(赠送源码数据库)JAVA、PHP,node.js,C++、python,大屏数据可视化等

SSM实验室耗材管理系统 摘 要 本课题研究的实验室耗材管理系统&#xff0c;主要功能模块包括用户管理、耗材管理、入库记录、出库记录、报废登记、供应商管理、耗材类别、实验室管理等&#xff0c;采取面对对象的开发模式进行软件的开发和硬体的架设&#xff0c;能很好的满足实…

【STM32 CubeMX】学STM必会的数据结构——环形缓冲区

文章目录 前言一、环形缓冲区是什么二、实现环形缓冲区实现分析2.1 环形缓冲区初始化2.2 写buf2.3 读buf2.4 测试 三、代码总况总结 前言 在嵌入式系统开发中&#xff0c;经常需要处理数据的缓存和传输&#xff0c;而环形缓冲区是一种常见且有效的数据结构&#xff0c;特别适用…

Codeforces Round 925 (Div. 3)

A. Recovering a Small String&#xff08;枚举&#xff09; 思路 每次枚举每一位 #include <iostream> using namespace std;int main() {int t;cin >> t;for (int i 0; i < t; i) {int n;cin >> n;if(n < 28){cout<<"aa"<<…

Rust 数据结构与算法:3栈:用栈实现符号匹配

1、符号匹配 如: (5+6)(7+8)/(4+3)、{ { ( [ ] [ ])}}、(a+b)(c*d)func() 等各类语句的符号匹配。 这里我们关注的不是数字而是括号,因为括号更改了操作优先级,限定了语言的语义,这是非常重要的。如果括号不完整,那么整个表达式就是错的。 括号都必须以成对匹配的形式出…

「数据结构」MapSet

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;Java数据结构 &#x1f387;欢迎点赞收藏加关注哦&#xff01; Map&Set &#x1f349;概念&#x1f349;模型&#x1f349;Map&#x1f34c;TreeMap和HashMap的区别&#x1f34c;Map常用方…