编程奇境:C++之旅,从新手村到ACM/OI算法竞赛大门(魔法帽:贪心思想)

本文主要是介绍编程奇境:C++之旅,从新手村到ACM/OI算法竞赛大门(魔法帽:贪心思想),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前面几期我们介绍了打怪的武器,但是刷怪的路上不能光凭蛮力,还要有智慧。需要有魔法帽的加持才能提升你的智慧点。

这期我们讲的是贪心思想。

什么是贪心呢?

贪心算法,就像是你肚子饿了,面对一桌子各式各样的美味点心,但妈妈说你只能拿一次,而且要尽可能地吃饱。怎么办呢?

你不会一个个去计算哪个组合能让你吃得最饱,那样太慢了。相反,你会用一个简单的方法:每次选择当前看起来最大的那个点心拿。比如,你先看到一个大蛋糕,就直接拿走,因为你知道蛋糕比小饼干能让你更快饱。就算之后你看到更大的点心,你也已经不能再换了,你只能做出在那一刻看起来最好的决定。

这就是贪心算法:在每个步骤中,都做出局部上最佳的选择,希望这样能带来全局上的最好结果。就像你每次都挑最大的点心,希望最后能吃得最饱。但是要注意,有时候这样不一定能得到全局最优解,比如如果后面有更大的点心你却已经拿了小的,就像生活中的一些决策,贪心策略可能让你错过一些更好的机会。但在某些特定问题里,贪心策略能高效地得到很好的解。

比如,你有不同面额的硬币要凑够一定的钱,贪心算法可能会让你每次都先用面额最大的硬币去凑,直到不够了再用次大的,这样往往能很快找到一个可行的凑钱方法,虽然不一定是所有可能中最少硬币的那个。

优点:

  1. 简单易懂:贪心算法的逻辑直接明了,容易理解和实现。就像小朋友做选择题时,每次选最明显的正确答案,一步步来。
  2. 运行效率高:因为它只关注当前的最佳选择,不需要考虑所有可能的解决方案,所以计算速度快,适合处理大量数据。
  3. 代码简洁:相较于其他复杂算法,贪心算法的代码通常更短,维护起来也更容易。
  4. 空间效率好:不需要存储大量中间结果,减少了对内存的需求。

缺点:

  1. 不一定得到最优解:虽然每一步都选最好的,但这些局部最优加在一起可能并不是全局最优。就像你选了每门课最喜欢的作业,但可能导致整个学期的成绩不是最好。
  2. 适用范围有限:不是所有问题都能用贪心算法解决,它最适合那些具有“贪心选择性质”的问题,即局部最优能导致全局最优的问题。
  3. 需要证明正确性:使用贪心算法前,往往需要严格证明这样做的每一步确实能导向最终的正确解,这有时很困难。
  4. 过早决定:一旦做出选择,就不可更改,可能导致错过了更好的解法,就像旅行时选了一条路,就不能回头尝试别的可能风景更美的路线。

总的来说,贪心算法在某些特定场景下非常有效,但必须小心应用,确保问题的特性允许局部最优解能导向全局最优解。

举个栗子

有 n 个人在一个水龙头前排队接水,假如每个人接水的时间为 Ti​,请编程找出这 n 个人排队的一种顺序,使得 n 个人的平均等待时间最小。

解析:

如果用暴力的思路:把每种组合都列一遍,看看哪种最小。这样的话时间太复杂了。

因为我们要让平均等待时间最小,相当于让总时间最小,我们观察到,总时间指的是前n-1个人接水的时间,因为第n个人接水的时候没有人等了。

那么是不是让接水最磨蹭的那个人排到最后去,这样大家就不用等太久了,这就是贪心,找到局部最优解。

所以我们对这n个人进行从小到大排序,时间越久的让ta越后面。

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define endl '\n'
int n;
double a[1005];
double sum[1005];
double ans;
int main()
{cin>>n;for(int i=1;i<=n;i++){cin>>a[i];sum[i]=sum[i-1]+a[i-1];//记录前i-1个人接水需要的时间 这是第i个人要等的 }sort(a+1,a+n+1);for(int i=2;i<=n;i++){ans+=sum[i];}cout<<ans/(n-1)<<endl;return 0;
}

再举个例子

小 A 有 𝑛个糖果盒,第 𝑖个盒中有 𝑎𝑖颗糖果。

小 A 每次可以从其中一盒糖果中吃掉一颗,他想知道,要让任意两个相邻的盒子中糖的个数之和都不大于 𝑥,至少得吃掉几颗糖。

解析:

因为是相邻的盒子,所以我们尽量贪心地去取可能会重复判断的盒子,那么就是相邻的盒子中靠后的那个,比如ABC,AB 和 BC 都出现了B,所以吃B盒子的是最优解。

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define endl '\n'
int n,x;
ll a[100005];
ll ans;
int main()
{cin>>n>>x;for(int i=1;i<=n;i++){cin>>a[i];}for(int i=2;i<=n;i++){if(a[i-1]+a[i]>x){int temp=a[i-1]+a[i]-x;//超过的部分 ans+=temp;//吃掉的糖果数 if(a[i]<temp)//如果这个盒子里的糖果不够吃了 {a[i]=0;temp-=a[i];a[i-1]-=temp;//就再吃前面盒子的 }else{a[i]-=temp;//够吃就直接吃 }}}cout<<ans<<endl;return 0;
}

练习题: 

P1223 排队接水 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

 P3817 小A的糖果 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

P1803 凌乱的yyy / 线段覆盖 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

P5019 [NOIP2018 提高组] 铺设道路 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

P1094 [NOIP2007 普及组] 纪念品分组 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

Problem - 1974A - Codeforces

Problem - 1925A - Codeforces

总结

贪心思想在算法竞赛中是一项非常非常重要的思想,它可以帮助我们大幅降低时间复杂度,甚至可能在O(1)的复杂度内就能解决问题,学会的方法嘛。。。就是多练!

百看不如一练,只有实践才是进步最快的方式,更要独立思考,如果想不出来了就看题解,会有眼前一亮的感觉。好啦,今天就到这里吧。下一期再见,记得给专栏点个关注,明天接着来哦~

这篇关于编程奇境:C++之旅,从新手村到ACM/OI算法竞赛大门(魔法帽:贪心思想)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1033741

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig