算法系列之贪心算法

在算法中,贪心算法(Greedy Algorithm)是一种常见的解决优化问题的算法。贪心算法的核心思想是:在每一步选择中都采取当前状态下最优的选择,即贪心的做出局部最优的决策,从而希望最终能够得到全局最优解。尽管贪心算法并不总是能够得到全局最优解,但在许多实际问题中,它能够提供足够好的解决方案,并且具有较高的计算效率。
本文将详细介绍贪心算法的基本概念、优缺点、实现步骤以及适用场景,并通过示例问题来展示贪心算法的应用。
基本概念
贪心算法它在每一步选择中都做出局部最优的选择,希望通过一系列局部最优的选择最终达到全局最优。贪心算法不会考虑过去的决策,而是一路向前的进行贪心的选择,不断缩小问题的范围,直至问题被解决。
我们通过找零问题来了解下贪心算法的工作原理:
问题描述:给定不同面额的硬币和一个总金额,要求用最少数量的硬币凑成该金额。
贪心策略:每次选择不大于且最接近剩余金额的最大的硬币,直到凑够总金额。
如图所示:我们需要找到凑够181元金额最少的硬币

优点及局限性
优点:
-
高效性:贪心算法通常具有较高的计算效率,适用于大规模问题。
-
简单性:贪心算法的实现通常较为简单,易于理解和编码。
缺点:
-
局部最优不一定全局最优:贪心算法并不总是能够得到全局最优解,有时只能得到近似解。
-
适用范围有限:贪心算法仅适用于具有贪心选择性质和最优子结构的问题。
示例:
比如币种有【1,20,50】,金额为60的话贪心算法只能找到 50+1*10 的兑换组合,有11张,而动态规划可以找到 20*3共3张的最优解。
贪心算法的实现步骤
贪心算法的实现通常包括以下几个步骤:
-
问题建模:将问题抽象为一个优化问题,明确目标函数和约束条件。
-
选择策略:确定每一步的局部最优选择策略。
-
迭代求解:从初始状态开始,逐步应用选择策略,直到达到终止条件。
-
验证解的有效性:验证最终得到的解是否满足问题的要求,并判断是否为全局最优解。
适用场景
- 最优化问题
问题需要找到最大值或最小值。
- 找零问题(用最少数量的硬币找零)。
- 最小生成树问题(Prim算法、Kruskal算法)。
- 最短路径问题(Dijkstra算法)。
- 区间问题
问题涉及区间选择或区间调度。
- 活动选择问题(选择最多的互不冲突的活动)。
- 区间覆盖问题(用最少的区间覆盖所有点)。
- 分配问题
问题涉及资源的分配或任务的调度。
- 背包问题(分数背包问题,贪心算法可解)。
- 任务调度问题(最小化完成时间或最大化任务数量)。
- 组合问题
问题涉及从一组元素中选择子集,满足某些条件。
- 霍夫曼编码(构建最优前缀编码)。
- 集合覆盖问题(近似解法)。
Java 实现示例
我们就用代码实现我们上边所示的找零问题,金额 161,面额【1,5,10,20,50,100】。代码如下:
/*** 贪心算法* 金额 161,面额[1,5,10,20,50,100],求最少的硬币个数*/
public class GreedyCoin {/*** 求解最少硬币个数* @param amount* @param coins* @return*/public static int minCoin(int amount, int[] coins) {//排序硬币面值Arrays.sort(coins);//初始化张数int count = 0;//循环面值,从大到小for (int i = coins.length - 1; i >= 0; i--) {count += amount / coins[i];amount = amount % coins[i];}// 如果金额没有完全凑齐,返回-1if(amount > 0){return -1;}return count;}public static void main(String[] args) {int amount = 161;int[] coins = {1,5,10,20,50,100};System.out.println(minCoin(amount, coins));}
}
示例题:观看⽂艺汇演
- 问题描述
某公园将举⾏多场⽂艺表演,很多演出都是同时进⾏,⼀个⼈只能同时观看⼀场演出,且不能迟到早退,由于演出分布在不同的演出场地,所以连续观看的演出最少有 15 分钟的时间间隔,⼩明是⼀个狂热的⽂艺迷,想观看尽可能多的演出。现给出演出时间表,请帮⼩明计算他最多能观看⼏场演出。
- 输⼊
第⼀⾏为⼀个数 N ,表⽰演出场数, 1 <= N <= 1000 。
接下来 N ⾏,每⾏两个空格分割的整数,第⼀个整数 T 表⽰演出的开始时间,第⼆个整数 L 表⽰
演出的持续时间, T 和 L 的单位为分钟, 0 <= T <= 1440, 0 < L <= 100 。
- 输出
最多能观看的演出场数。
- 示例
示例1
输入
2
720 120
840 120
输出
1
示例2
输入
2
20 60
100 60
输出
2
示例3
输入
4
10 20
100 20
150 60
80 40
输出
3
- 问题分析
我们可以储存每⼀场演出的开始和结束时间,即按照 [start, end] 的⽅式进⾏储存。由于题⽬要求每间隔 15 分钟才能够看下⼀场演出,所以我们可以把每⼀场演出的结束时间再加上 15 分钟,这样题⽬就转变为:考虑所有不重叠的 [start, end] 区间的最⼤数⽬。
我们按开始时间对这些区间进行排序, 情况1:演出2的开始时间在演出1之后,就可以看完演出1再看演出2; 情况2:演出2的开始时间在演出1的期间,结束时间在演出1的结束时间之后,看完演出1无法看演出2 情况3:演出2的结束时间在演出1的结束时间之前,没必要看演出1。
- 代码实现
/*** 文艺演出类*/
public class Performance implements Comparable<Performance>{int start; // 演出开始时间int end; // 演出结束时间public Performance(int start, int end) {this.start = start;this.end = end;}@Overridepublic int compareTo(Performance other) {return this.start - other.start; // 按开始时间从小到大排序}
}/*** 观看⽂艺汇演贪心算法
*/
public class GreedyPerformance {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int N = scanner.nextInt();List<Performance> performances = new ArrayList<>();for (int i = 0; i < N; i++) {int start = scanner.nextInt();int during = scanner.nextInt();int end = start + during + 15;performances.add(new Performance(start, end));}Collections.sort(performances);int count = 0; // 观看演出的场数int lastEnd = 0; // 上一场演出的结束时间,初始化为0for(Performance performance:performances){int start = performance.start;int end = performance.end;// 如果当前演出的开始时间大于上一场演出的结束时间,则观看if(start >= lastEnd){count++;lastEnd=end;// 如果当前演出的开始时间小于上一场演出的结束时间,当前演出的结束时间大于上一场的结束时间,则不能观看}else if(start < lastEnd && lastEnd<= end){continue;// 如果当前演出的结束时间小于上一场的结束时间,则观看当前场}else if(lastEnd > end){lastEnd=end;}}System.out.println(count);}}
总结
贪心算法是一种简单而高效的优化策略,适用于许多实际问题。尽管它并不总是能够得到全局最优解,但在许多情况下,贪心算法能够提供足够好的解决方案,并且具有较高的计算效率。理解贪心算法的基本原理和适用场景,能够帮助我们在实际问题中更好地应用这一策略。
相关文章:
算法系列之贪心算法
在算法中,贪心算法(Greedy Algorithm)是一种常见的解决优化问题的算法。贪心算法的核心思想是:在每一步选择中都采取当前状态下最优的选择,即贪心的做出局部最优的决策,从而希望最终能够得到全局最优解。尽…...
将产品照片(form.productPhotos)转为 JSON 字符串发送给后端
文章目录 1. 前端 form.productPhotos 的当前处理a. 组件绑定b. 当前发送逻辑 2. 如何将 form.productPhotos 转为 JSON 字符串发送给后端a. 修改前端 save() 方法b. 确保 esave API 支持接收字符串 基于你提供的 identify-form.vue 代码,我将分析如何将产品照片&a…...
『大模型笔记』详细对比GraphRAG与传统RAG!
详细对比GraphRAG与传统RAG! 文章目录 详细对比GraphRAG与传统RAG!要点最终内容1. GraphRAG的作用与应用场景2. GraphRAG与传统RAG的对比3. GraphRAG的工作原理4. GraphRAG如何提高准确性和提供完整答案5. GraphRAG在开发和维护中的优势6. GraphRAG对生产环境的影响7. GraphR…...
安全面试3
文章目录 一个单位的一级域名可能不止一个,怎么收集某个单位的所有域名,注意不是子域名用转义字符防御时,如果遇到数据库的列名或是表名本身就带着特殊字符,应该怎么做宽字节注入原理防御宽字节注入的方法 基于黑白名单的修复&…...
软件测试:1、单元测试
1. 单元测试的基本概念 单元(Unit):软件系统的基本组成单位,可以是函数、模块、方法或类。 单元测试(Unit Testing):对软件单元进行的测试,验证代码的正确性、规范性、安全性和性能…...
球队训练信息管理系统设计与实现(代码+数据库+LW)
摘 要 传统办法管理信息首先需要花费的时间比较多,其次数据出错率比较高,而且对错误的数据进行更改也比较困难,最后,检索数据费事费力。因此,在计算机上安装球队训练信息管理系统软件来发挥其高效地信息处理的作用&a…...
【Bluedroid】AVRCP 连接源码分析(二)
接着上一篇【Bluedroid】AVRCP 连接源码分析(一)-CSDN博客,继续AVRCP连接的源码分析。 getcapabilities_cmd packages/modules/Bluetooth/system/btif/src/btif_rc.cc /***************************************************************************** Function …...
OSS(对象存储服务)
OSS(对象存储服务) 是一种用于存储和管理非结构化数据的云存储服务,其核心设计面向海量数据的高扩展性、高可靠性和低成本存储。以下从定义、核心原理、架构特点和应用场景等方面详细介绍: 一、什么是OSS? OSS&#x…...
《深入理解JVM》实战笔记(二): 类加载机制与类加载器
序言 Java 语言的强大之处之一在于其动态加载的能力,使得 Java 程序可以在运行时加载新的类,而不需要在编译时确定所有的类信息。这一切都离不开 JVM 的类加载机制。本篇博客将详细探讨 JVM 的类加载过程以及类加载器的工作原理,帮助你更深入…...
ChromeDriver下载
平时为了下个驱动,到处找挺麻烦,收集了很多无偿分享给需要的人,仅供学习和交流。 ChromeDriver 102.0.5005.61 ChromeDriver 105.0.5195.102 ChromeDriver 108.0.5359.71 ChromeDriver 111.0.5563.64 ChromeDriver 116.0.5845.97 Chrom…...
《深度学习实战》第1集:深度学习基础回顾与框架选择
本专栏系列博文旨在帮助读者从深度学习的基础知识逐步进阶到前沿技术,涵盖理论、实战和行业应用。每集聚焦一个核心知识点,并结合实际项目进行实践,避免空谈理论,简洁明快,快速切入代码,所有代码都经过验证…...
Docker 部署AnythingLLM
两个指令搞定 1.下载镜像 docker pull mintplexlabs/anythingllm 2.运行容器 export STORAGE_LOCATION$HOME/anythingllm mkdir -p $STORAGE_LOCATION chmod -R 777 $STORAGE_LOCATION touch "$STORAGE_LOCATION/.env" docker run -d -p 3001:3001 \ --cap-add SY…...
泰山派RK3566移植QT,动鼠标时出现屏幕闪烁
总结: 交叉编译到 泰山派rk3566跑调海康摄像头的qt应用程序失败了。 X11无效窗口。 移植QT注意 屏幕分辨率不要改。改了执行QT的时候,framebuffer识别不出设备。 命令行安装QT-Creator sudo install 类似的指令安装Qt-Creator时,可能找不到编…...
关于Java 反射的简单易懂的介绍
目录 #0.总览 #1. 类的反射 ①介绍 ②获取 ③作用 获取构造函数: 创建实例: 字段操作: 方法操作: 获取修饰符: #2.总结 #0.总览 反射,官方是这样介绍它的: Reflection is a …...
市场趋势中突破确认的多维度判断方法
波动率突破策略是众多交易者广泛采用的重要交易策略之一。而在这一策略中,准确判断突破是否有效,是决定交易成败的关键环节。仅仅依据单一因素来确认突破,往往会使交易者陷入误判的困境,导致不必要的损失。因此,采用多…...
网络空间安全(2)应用程序安全
前言 应用程序安全(Application Security,简称AppSec)是一个综合性的概念,它涵盖了应用程序从开发到部署,再到后续维护的整个过程中的安全措施。 一、定义与重要性 定义:应用程序安全是指识别和修复应用程序…...
【MyBatis】CRUD、配置解析、ResultMap、分页实现
目录标题 1、Mybatis简介1.1、什么是MyBatis1.2、持久化1.3、持久层1.4、为什么需要MybatisMyBatis的优点 2.1、代码演示搭建实验数据库导入MyBatis相关 jar 包 03、CRUD操作3.1、namespace3.2、select3.3、insert3.4、update3.5、delete 04、MyBatis配置解析4、配置解析4.3、m…...
Linux系统编程之高级信号处理
概述 在前一篇文章中,我们介绍了signal函数、sigaction函数等基本的信号处理方法。在本篇中,我们将介绍信号处理的一些高级用法,包括:阻塞与解除阻塞、定时器等。 阻塞与解除阻塞 有时候,我们不希望某个信号立即被处理…...
深度学习驱动的车牌识别:技术演进与未来挑战
一、引言 1.1 研究背景 在当今社会,智能交通系统的发展日益重要,而车牌识别作为其关键组成部分,发挥着至关重要的作用。车牌识别技术广泛应用于交通管理、停车场管理、安防监控等领域。在交通管理中,它可以用于车辆识别、交通违…...
钉钉快捷免登录 通过浏览器打开第三方系统,
一、钉钉内跳转至浏览器的实现 使用钉钉JSAPI的跳转接口 在钉钉内通过dd.biz.navigation.openLink方法强制在系统浏览器中打开链接。此方法需在钉钉开发者后台配置应用权限,确保应用具备调用该API的资格37。 示例代码: dd.ready(() > {dd.biz.navigat…...
Flask RESTful 示例
目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题: 下面创建一个简单的Flask RESTful API示例。首先,我们需要创建环境,安装必要的依赖,然后…...
OpenLayers 分屏对比(地图联动)
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 地图分屏对比在WebGIS开发中是很常见的功能,和卷帘图层不一样的是,分屏对比是在各个地图中添加相同或者不同的图层进行对比查看。…...
【生成模型】视频生成论文调研
工作清单 上游应用方向:控制、速度、时长、高动态、多主体驱动 类型工作基础模型WAN / WAN-VACE / HunyuanVideo控制条件轨迹控制ATI~镜头控制ReCamMaster~多主体驱动Phantom~音频驱动Let Them Talk: Audio-Driven Multi-Person Conversational Video Generation速…...
【堆垛策略】设计方法
堆垛策略的设计是积木堆叠系统的核心,直接影响堆叠的稳定性、效率和容错能力。以下是分层次的堆垛策略设计方法,涵盖基础规则、优化算法和容错机制: 1. 基础堆垛规则 (1) 物理稳定性优先 重心原则: 大尺寸/重量积木在下…...
Modbus RTU与Modbus TCP详解指南
目录 1. Modbus协议基础 1.1 什么是Modbus? 1.2 Modbus协议历史 1.3 Modbus协议族 1.4 Modbus通信模型 🎭 主从架构 🔄 请求响应模式 2. Modbus RTU详解 2.1 RTU是什么? 2.2 RTU物理层 🔌 连接方式 ⚡ 通信参数 2.3 RTU数据帧格式 📦 帧结构详解 🔍…...
二维FDTD算法仿真
二维FDTD算法仿真,并带完全匹配层,输入波形为高斯波、平面波 FDTD_二维/FDTD.zip , 6075 FDTD_二维/FDTD_31.m , 1029 FDTD_二维/FDTD_32.m , 2806 FDTD_二维/FDTD_33.m , 3782 FDTD_二维/FDTD_34.m , 4182 FDTD_二维/FDTD_35.m , 4793...
MeshGPT 笔记
[2311.15475] MeshGPT: Generating Triangle Meshes with Decoder-Only Transformers https://library.scholarcy.com/try 真正意义上的AI生成三维模型MESHGPT来袭!_哔哩哔哩_bilibili GitHub - lucidrains/meshgpt-pytorch: Implementation of MeshGPT, SOTA Me…...
基于谷歌ADK的 智能产品推荐系统(2): 模块功能详解
在我的上一篇博客:基于谷歌ADK的 智能产品推荐系统(1): 功能简介-CSDN博客 中我们介绍了个性化购物 Agent 项目,该项目展示了一个强大的框架,旨在模拟和实现在线购物环境中的智能导购。它不仅仅是一个简单的聊天机器人,更是一个集…...
vxe-table vue 表格复选框多选数据,实现快捷键 Shift 批量选择功能
vxe-table vue 表格复选框多选数据,实现快捷键 Shift 批量选择功能 查看官网:https://vxetable.cn 效果 代码 通过 checkbox-config.isShift 启用批量选中,启用后按住快捷键和鼠标批量选取 <template><div><vxe-grid v-bind"gri…...
VUE3 ref 和 useTemplateRef
使用ref来绑定和获取 页面 <headerNav ref"headerNavRef"></headerNav><div click"showRef" ref"buttonRef">refbutton</div>使用ref方法const后面的命名需要跟页面的ref值一样 const buttonRef ref(buttonRef) cons…...
