数据结构与算法(三)贪心算法(Java)

目录

    • 一、简介
      • 1.1 定义
      • 1.2 基本步骤
      • 1.3 优缺点
    • 二、经典示例
      • 2.1 选择排序
      • 2.2 背包问题
    • 三、经典反例:找零钱
      • 3.1 题目
      • 3.2 解答
      • 3.3 记忆化搜索实现
      • 3.4 动态规划实现

一、简介

1.1 定义

贪心算法(Greedy Algorithm),又名贪婪法,是寻找 最优解问题 的常用方法。

  • 将求解过程 分成若干个步骤,每个步骤都应用贪心原则,选取当前状况下最好/最有的选择(局部最有利的选择),并以此希望最后堆叠出的结果也是最好/最优的解。

1.2 基本步骤

  • 步骤1:从某个初始解出发;
  • 步骤2:把求解的问题分成若干个子问题;
  • 步骤3:对每一子问题求解,得到子问题的局部最优解;
  • 步骤4:把子问题的局部最优解合成原来问题的一个解。

1.3 优缺点

优点:

  • 简单、高效,省去为了寻找最优解可能需要穷举的操作,通常作为其他算法的辅助算法来使用

缺点:

  • 不从整体上考虑其它可能情况,每次选取局部最优解,不再进行回溯处理,所以 并非一定能得到整体最优解

二、经典示例

2.1 选择排序

没错,我们常见的选择排序就是运用了贪心算法的思想。

题目:

  • 实现数字数组递增排序。

解答:

从数组的零下标开始,依次从后面找到最小的元素下标与当前位置的元素互换,这个在后面寻找最小元素的过程就是贪心的思想。

贪心策略寻找最小的元素,(贪心地)认定此元素就是当前位置的最小元素,然后遍历每一个位置。

public void choiceSort(int[] arr) {for (int i = 0; i < arr.length; i++) {int minIndex = i;for (int j = i + 1; j < arr.length; j++) {minIndex = arr[j] < arr[minIndex] ? j : minIndex;}if (minIndex != i) {int tmp = arr[i];arr[i] = arr[minIndex];arr[minIndex] = tmp;}}
}

2.2 背包问题

题目:

有一个背包,容量由你自己输入,有n个物品,每个物品都具有容量与价值,这些都是由你自己输入的,请问,要怎么放物品到背包里,才能使得总价值最大呢,放入背包的总容量要小于等于背包的总容量。(如果一个物品放不下,则可以拆分成多个小块)
背包:M:100
物品:N:7
重量 价值
10 20
20 40
30 30
25 20
50 40
10 35
60 70

解答:

每个物品都具有自己的重量与价格,不妨计算出每个物品的单位价值。

  • 单位价值: 价值/重量,即每份重量的价值。

然后我们将这些物品 按照单位价值递减排序。这样一来就简单了,只需用贪心算法,依次把最大单位价值的物品价值和重量相加 就行了。

贪心策略:单位价值最大的物品,我们假设它就是最好的,直接把它放在背包里面。

public static void main(String[] args) {int[][] items = new int[7][2];items[0][0] = 10; items[0][1] = 20;items[1][0] = 20; items[1][1] = 40;items[2][0] = 30; items[2][1] = 30;items[3][0] = 25; items[3][1] = 20;items[4][0] = 50; items[4][1] = 40;items[5][0] = 10; items[5][1] = 35;items[6][0] = 60; items[6][1] = 70;int capacity = 100;System.out.println("背包的容量:" + capacity);StringBuilder builder = new StringBuilder();for (int[] item : items) {builder.append(Arrays.toString(item));}System.out.println(items.length + " 个物品的重量、价值:" + builder.toString());int maxValue = maxValue(items, capacity);System.out.println("最大价值:" + maxValue);
}public static int maxValue(int[][] items, int capacity) {// 计算单位价值double[] prices = new double[items.length];Map<Double, List<Integer>> positionMap = new HashMap<>(items.length);for (int i = 0; i < items.length; i++) {prices[i] = 1.0 * items[i][1] / items[i][0];List<Integer> positions = positionMap.getOrDefault(prices[i], new ArrayList<>());positions.add(i);positionMap.put(prices[i], positions);}// 排序Arrays.sort(prices);int weight = 0;int maxValue = 0;for (int i = prices.length - 1; i >= 0; i--) {List<Integer> positions = positionMap.get(prices[i]);if (positions != null) {Integer position = positions.remove(0);if (positions.size() == 0) {positionMap.remove(prices[i]);}if (weight + items[position][1] < capacity) {weight += items[position][0];maxValue += items[position][1];System.out.println("重量为 " + items[position][0] + ",价值为 " + items[position][1] + " 的物品被放入背包,剩余容量:" + (capacity - weight));}}}return maxValue;
}

执行结果:

在这里插入图片描述


三、经典反例:找零钱

322. 零钱兑换

3.1 题目

假设你开了间小店,不能电子支付,钱柜里的货币只有 25 分、20分、10 分、5 分和 1 分 四种硬币,如果你是售货员且要找给客户 41 分钱的硬币,如何安排才能找给客人的钱既 正确 且硬币的个数又 最少

3.2 解答

我们看到这种题目可能第一个想法就是用 贪心算法 进行解决,其实不然,由于贪心算法不能进行回溯处理,所以并不能取得最优解。

  • 41 分钱按照贪心算法,先找 25分,剩余 16分,再找 10分、5分、1分,共需要 4 枚硬币。
  • 实际情况下,我们只需要找两个 20分钱,再找一个 1分钱就够了,共需要 3 枚硬币。

那么不用贪心算法,应该用什么算法呢?其实有两种方法可以解决:

  • 一是 贪心+回溯,即 记忆化搜索
  • 二是 动态规划

3.3 记忆化搜索实现

  • 记忆化搜索实现方式,就是自顶向下遍历,先查用完一枚硬币之后还剩多少钱,再根据剩余的钱进行迭代。

代码实现需要注意以下几点:

  • int[] 类型数组初始化值为0;
  • 针对需要记忆的组合,不光要记忆成功的情况,失败的情况也要记录。
class Solution {public static void main(String[] args) {int amount = 41;int[] arr1 = {1, 5, 10, 20, 25};System.out.println("零钱总数为:" + amount);System.out.println("硬币面值为:" + Arrays.toString(arr1));int result1 = coinChange(arr1, amount);System.out.println("最少使用硬币数:" + result1);}public static int coinChange(int[] coins, int amount) {if (amount == 0) {return 0;}return handleCoin(coins, new int[amount], amount);}private static int handleCoin(int[] coins, int[] his, int coinAmount) {if (coinAmount < 0) {return -1;}if (coinAmount == 0) {return 0;}if (his[coinAmount - 1] != 0) {return his[coinAmount - 1];}int minCount = Integer.MAX_VALUE;for (int coin : coins) {int tmpMinCount = handleCoin(coins, his, coinAmount - coin);if (tmpMinCount != -1 && tmpMinCount + 1 < minCount) {minCount = tmpMinCount + 1;}}his[coinAmount - 1] = minCount == Integer.MAX_VALUE ? -1 : minCount;return his[coinAmount - 1];}
}

执行结果:

在这里插入图片描述

3.4 动态规划实现

  • 动态规划实现,则是自底向上,先计算从1开始每个金额所需的最小零钱数,直到找到需要的钱数,过程中对于剩余钱数所需要的最小零钱数可以直接使用前面计算好的数据。

代码实现需要注意以下几点:

  • 对于数值类型的映射,不要用 Map 类型,用 int[] 类型效率更高;
  • 在循环迭代的过程中,只需要加硬币数就可以了,不用再去迭代考虑其他的组合。
class Solution {public static void main(String[] args) {int amount = 41;int[] arr1 = {1, 5, 10, 20, 25};System.out.println("零钱总数为:" + amount);System.out.println("硬币面值为:" + Arrays.toString(arr1));int result1 = coinChange(arr1, amount);System.out.println("最少使用硬币数:" + result1);}public static int coinChange(int[] coins, int amount) {// k-零钱和,v-最小零钱量int[] his = new int[amount + 1];Arrays.fill(his, amount + 1);his[0] = 0;for (int i = 1; i <= amount; i++) {for (int coin : coins) {if (coin <= i) {his[i] = Math.min(his[i], his[i - coin] + 1);}}}return his[amount] == amount + 1 ? -1 : his[amount];}
}

执行结果:

在这里插入图片描述

整理完毕,完结撒花~ 🌻





参考地址:

1.小白带你学—贪心算法(Greedy Algorithm),https://zhuanlan.zhihu.com/p/53334049

2.贪心算法思想详解+示例代码,https://blog.csdn.net/jj6666djdbbd/article/details/126971331

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

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

相关文章

SAP Smartform小结

SAP系统做打印单据用的, 感觉很不好用, 特别是要嵌入韩文时必须使用嵌入的word编辑器,运行速度简直不可忍受. 见过一些Adobe interactive form的示例, 看着相当不错, 不过据说需要花money额外买licence, 哪有smartform这种免费东西来得实惠. 一般打印需求,会要求有标题抬头,打…

Java中有几种基本数据类型以及转换方式【Java面经(1)】

问&#xff1a;Java中有几种基本数据类型呢&#xff1f;以及它们之间的转换方式。详细介绍下 总共有8种基本数据类型 byte 、short 、long 、float 、double 、boolean 、char 详细类型以及字节数&#xff1a; 基本数据类型的转换方式 自动类型转换&#xff1a;小–>大 byt…

MyBatis的解析和运行原理

文章目录 MyBatis的解析和运行原理MyBatis的工作原理 MyBatis的解析和运行原理 MyBatis编程步骤是什么样的&#xff1f; 1、 创建SqlSessionFactory 2、 通过SqlSessionFactory创建SqlSession 3、 通过sqlsession执行数据库操作 4、 调用session.commit()提交事务 5、 调用…

[数据结构]-红黑树

前言 作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正 目录 一、红黑树的…

身在苹果心系Linux:Linux基金会主管在开源峰会使用macOS做演示

2017开源峰会正在进行中&#xff0c;来自全球各地的Linux大厂和开源大厂纷纷涌向洛杉矶参加这次活动。 Linux基金会执行董事Jim Zemlin在推文中宣布&#xff1a;2017年是属于Linux桌面的一年&#xff01; ▲图片来源&#xff1a;itsfoss.com 然而&#xff0c;据外媒itsfoss报…

【开源】基于Vue和SpringBoot的数字化社区网格管理系统

项目编号&#xff1a; S 042 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S042&#xff0c;文末获取源码。} 项目编号&#xff1a;S042&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块三、开发背景四、系统展示五、核心源码5…

计算器的模拟实现

计算器的模拟实现 一、实验题目&#xff1a;计算器二&#xff1a;实验目的&#xff1a;三&#xff1a;实验内容与实现1&#xff1a;【实验内容】2&#xff1a;【实验实现】1.计算器界面的实现&#xff0c;如下图所示&#xff1a;2&#xff1a;各项功能的实现&#xff0c;如下图…

JVM——垃圾回收(方法区中的垃圾回收和(堆回收)自动垃圾回收)

目录 1.自动垃圾回收介绍1.C/C的内存管理2.Java的内存管理3.垃圾回收的对比 2.方法区的回收方法区的回收 – 手动触发回收 3.堆回收1.引用计数法2.可达性分析算法 1.自动垃圾回收介绍 1.C/C的内存管理 ⚫ 在C/C这类没有自动垃圾回收机制的语言中&#xff0c;一个对象如果不再…

过渡曲线的构造之平面PH曲线

平面PH曲线的构造及其相应性质 平面PH曲线的构造及其相应性质PH曲线理论三次PH曲线的构造及性质四次PH曲线的构造及性质五次PH曲线的构造及性质非尖点五次PH曲线尖点五次PH曲线 参考文献 平面PH曲线的构造及其相应性质 过渡曲线常需要满足在连接点处位置连续、曲率连续以及切线…

韵达快递查询,韵达快递单号查询,一键筛选出单号中的退回件

批量查询韵达快递单号的物流信息&#xff0c;并将其中的退回件一键筛选出来。 所需工具&#xff1a; 一个【快递批量查询高手】软件 韵达快递单号若干 操作步骤&#xff1a; 步骤1&#xff1a;运行【快递批量查询高手】软件&#xff0c;第一次使用的朋友记得先注册&#xff…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《基于多尺度分量特征学习的用户级超短期负荷预测》

这篇文章的标题表明研究的主题是用户级超短期负荷预测&#xff0c;并且该预测方法基于多尺度分量特征学习。让我们逐步解读这个标题&#xff1a; 用户级&#xff1a; 这表示研究的焦点是在个体用户层面上进行的。负荷预测可能是指电力系统中的负荷&#xff0c;即电力需求。用户…

队列详解(C语言实现)

文章目录 写在前面1 队列的定义2 队列的初始化3 数据入队列4 数据出队列5 获取队头元素6 获取队尾元素7 获取队列元素个数8 判断队列是否为空8 队列的销毁 写在前面 本片文章详细介绍了另外两种存储逻辑关系为 “一对一” 的数据结构——栈和队列中的队列&#xff0c;并使用C语…