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

镇江网站建设报价建筑云平台

镇江网站建设报价,建筑云平台,网络营销推广实例,网站建设的制度建设文章目录 前置知识860.柠檬水找零题目描述解题思路代码 406.根据身高重建队列题目描述解题思路代码 452. 用最少数量的箭引爆气球题目描述踩坑-进行模拟正确思路的贪心 总结 前置知识 参考前文 参考文章: LeetCode刷题笔记【23】:贪心算法专题-1&#x…

文章目录

  • 前置知识
  • 860.柠檬水找零
    • 题目描述
    • 解题思路
    • 代码
  • 406.根据身高重建队列
    • 题目描述
    • 解题思路
    • 代码
  • 452. 用最少数量的箭引爆气球
    • 题目描述
    • 踩坑-进行模拟
    • 正确思路的贪心
  • 总结

前置知识

参考前文

参考文章:
LeetCode刷题笔记【23】:贪心算法专题-1(分发饼干、摆动序列、最大子序和)
LeetCode刷题笔记【24】:贪心算法专题-2(买卖股票的最佳时机II、跳跃游戏、跳跃游戏II)
LeetCode刷题笔记【25】:贪心算法专题-3(K次取反后最大化的数组和、加油站、分发糖果)

860.柠檬水找零

题目描述

截图

LeetCode链接:https://leetcode.cn/problems/lemonade-change/description/

解题思路

思路: 用vector<int> counter(3,0)来记录5, 10, 20元钞票的数量;
如果顾客正好给5 , ‘ c o u n t e r [ 0 ] + + ‘ ; 如果顾客给的钱 ‘ m > 5 ‘ , `counter[0]++`; 如果顾客给的钱`m>5` ,counter[0]++;如果顾客给的钱m>5‘, target = m-5;
m=15, m=5的时候分类讨论即可;
当发现counter[0]<0时返回false;
最后返回true

代码

class Solution {
public:bool lemonadeChange(vector<int>& bills) {vector<int> counter(3,0);for(int m : bills){int target = m-5;if(target==0){//1 顾客直接给5$counter[0]++;}else if(target==5){//2 顾客给10$counter[1]++;counter[0]--;}else if(target==15){//3 顾客给20$if(counter[1]>=1){//3.1 有10$counter[2]++;counter[1]--;counter[0]--;}else{//3.2 没有10$counter[2]++;counter[0] -= 3;}}if(counter[0]<0 || counter[1]<0)return false;}return true;}
};

406.根据身高重建队列

题目描述

截图

LeetCode链接:https://leetcode.cn/problems/queue-reconstruction-by-height/description/

解题思路

先按照身高, 进行从大到小排列, 身高相同的人根据k, 从小到大排列;
然后从排列后的people数组中依次提取person, 加入ans;
加入时直接通过k, 选择空位插入;

感觉似乎有些玄学, 如果一定要总结的话, 应该着眼于sort之后插入的环节:
每次插入的这个P都是未插入的person里面最高的, 相比于已经排好队的人, 是更矮的, 所以只要从前往后数k个, 直接插入即可.

代码

class Solution {
public:vector<vector<int>> reconstructQueue(vector<vector<int>>& people) {sort(people.begin(), people.end(),[](vector<int>& a, vector<int>& b){return (a[0]>b[0]) || (a[0]==b[0] && a[1]<b[1]);});vector<vector<int>> ans;for(vector<int> person : people){ans.insert(ans.begin()+person[1], person);}return ans;}
};

452. 用最少数量的箭引爆气球

题目描述

在这里插入图片描述

LeetCode链接:https://leetcode.cn/problems/minimum-number-of-arrows-to-burst-balloons/description/

踩坑-进行模拟

思路: 创建一个unordered_map<int,int> counter, 记录从x坐标垂直向上看, 有多少个气球
每次都选择气球最多的那个x坐标发射一支箭, 然后看击破哪些气球, 更新counter
直到气球被打完
思考了一下, 还是用vector<int> counter吧, 先遍历一下points, 求一下x轴最大值

class Solution {
private:vector<int> refreshX(vector<vector<int>>& points, int maxX){vector<int> counter(maxX+1, 0);for(vector<int> point : points){for(int x=point[0]; x<=point[1]; ++x){counter[x]++;}}return counter;}
public:int findMinArrowShots(vector<vector<int>>& points) {if(points.size()==0)return 0;else if(points.size()==1)return 1;int maxX=INT_MIN;for(vector<int> point : points){maxX = max(maxX, point[1]);}vector<int> counter = refreshX(points, maxX);// for(int i=0; i<counter.size(); ++i){//     cout << i << ":" << counter[i] << " " << endl;// }int ans=0;while(!points.empty()){ans ++;// 没有跳出, 那么本轮一定要射出一箭// 寻找本轮需要在哪个位置(shootingX)射箭int shootingX=0, shootingNum=INT_MIN;for(int i=1; i<counter.size(); ++i){if(counter[i] > shootingNum){shootingNum = counter[i];shootingX = i;}}for(int i=0; i<points.size(); ++i){points.erase(remove_if(points.begin(), points.end(), [shootingX](vector<int> p){return p[0]<=shootingX && p[1]>=shootingX;}), points.end());}counter = refreshX(points, maxX);}return ans;}
};

以上写法没问题, 但是没有考虑区间为负的情况
这样的话咱们还是用unordered_map

class Solution {
private:map<int,int> refreshX(vector<vector<int>>& points){map<int,int> counter;for(vector<int> point : points){for(int x=point[0]; x<=point[1]; ++x){counter[x]++;}}return counter;}
public:int findMinArrowShots(vector<vector<int>>& points) {if(points.size()==0)return 0;else if(points.size()==1)return 1;bool overlapping = false;for(int i=0; i<points.size()-1; ++i){if(points[i][1]>=points[i+1][0])overlapping=true;}if(overlapping==false)return points.size();map<int,int> counter = refreshX(points);int ans=0;while(!points.empty()){ans ++;// 没有跳出, 那么本轮一定要射出一箭// 寻找本轮需要在哪个位置(shootingX)射箭// cout << "此时的counter情况是: " ;// for(auto& pair : counter){//     cout << pair.first << ":" << pair.second << " " ;// }// cout << endl;int shootingX=0, shootingNum=INT_MIN;for(auto& pair : counter){if(pair.second > shootingNum){shootingNum = pair.second;shootingX = pair.first;}}// cout << "shootingX= " << shootingX << endl;for(int i=0; i<points.size(); ++i){points.erase(remove_if(points.begin(), points.end(), [shootingX](vector<int> p){return p[0]<=shootingX && p[1]>=shootingX;}), points.end());}counter = refreshX(points);}return ans;}
};

正确思路的贪心

以上想法很好, 也可以通过大部分案例, 就是每次射爆最多的气球;
但是对于测试用例[[9,17],[4,12],[4,8],[4,8],[7,13],[3,4],[7,12],[9,15]]而言
你先从x=8/9/10处射箭(最开始时这三点重叠气球最多), 之后就需要再射2
但是如果第一箭先x=4处射, 那么之后只用射1

所以转变思路:
① 先用左区间为index, sort points
依次从第二个气球i开始遍历, 不断更新"重叠的一组气球";
如果气球ii-1没有重叠, 那么ans++;
否则就更新i的右边界为ii-1的最小右边结(which means是"这一组重叠气球的右边界")

class Solution{
public:int findMinArrowShots(vector<vector<int>>& points){if(points.empty())return 0;sort(points.begin(), points.end(), [](vector<int>& a, vector<int>&b){return a[0] < b[0];});int ans=1;for(int i=1; i<points.size(); ++i){if(points[i][0] > points[i-1][1]){ans ++;}else{points[i][1] = min(points[i][1], points[i-1][1]);}}return ans;}
};

总结

贪心真的防不胜防, 波云诡谲, 难以捉摸;
今天第三题本来以为自己已经找到正确的贪心思路了(每次都捡能打掉最多气球的点射箭), 然而并不是;
所以个人其实认为将这些乱七八糟的东西都归到"贪心算法"中进行分类, 某种程度上并不是很严谨合理.
做的过程中多看看题解, 学习参考为主吧, 别硬磕, 伤身劳心费神.

本文参考:
柠檬水找零
根据身高重建队列
用最少数量的箭引爆气球

http://www.yayakq.cn/news/76979/

相关文章:

  • 成品网站源码的优化技巧淄博做网站的公司
  • 谷歌没收录网站主页 301重定向短视频推广策划方案
  • 可信网站的作用大连品尚茗居装修公司怎么样
  • 什么网站可以做海报网页制作与设计的英文
  • 社交网站盈利吗网站开发模板专家数据库
  • 邢台网站建设策划关键词优化搜索排名
  • 蚌埠市建设银行官方网站做网站赚钱需要多少人手
  • 南京润盛建设集团有限公司网站免费建网站模板
  • 专做火影黄图的网站有谁用2008做网站服务器
  • 网站音乐播放器插件西维科技做网站怎么样
  • VPS如何做镜像网站保险网站
  • 网站建设中问题分析与解决免费的制作网站
  • 提出网站推广途径和推广要点旅游网站内容规划
  • 商企在线营销型网站在线网站推广工具
  • 站长工具网址查询安徽省建设厅网站职称
  • 网站建设免费模版杭州专业网站设计制作公司
  • 服装网站建设多少钱电商运营模式
  • 学校网站平台建设方案怎么申请做网站
  • 网站 功能需求学校宣传软文
  • 免费的网站推广渠道装饰公司网站如何布局
  • 领导高度重视门户网站建设网站建设问卷
  • 网站整合营销河北建设工程信息网中标公示
  • 营销型网站建设公司是干嘛的大网站
  • 临沂网站建设举措wordpress支付下载插件
  • 网站上的个人词条怎么做的营销型网站建设ppt模板下载
  • 台州手机网站开发企业信息化建设网站
  • 信阳网站建设培训网站标题优化工具
  • 网站开发研究现状公司简单网站多少钱
  • 线上网站设计找外包公司做网站价钱
  • 丽江建设信息网站新版wordpress