编程奇境: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

相关文章

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

C#异步编程ConfigureAwait的使用小结

《C#异步编程ConfigureAwait的使用小结》本文介绍了异步编程在GUI和服务器端应用的优势,详细的介绍了async和await的关键作用,通过实例解析了在UI线程正确使用await.Conf... 异步编程是并发的一种形式,它有两大好处:对于面向终端用户的GUI程序,提高了响应能力对于服务器端应

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析