贪心算法(Greedy Algorithm)是一种在每一步选择中都采取在当前状态下最好或最优(即最有利)的选择,从而希望导致结果是全局最好或最优的算法。贪心算法通常用于解决优化问题,如最小化成本、最大化收益等。然而,贪心算法并不总是能够得到全局最优解,但它具有直观、高效、易于实现等优点,因此在许多实际问题中得到了广泛应用。
基本思想
贪心算法实现步骤
贪心算法的适用范围
局限性
优化
题目:
某天早上公司领导找你解决一个问题,明天公司有N个同等级的会议需要使用同一个会议室,现在给你这个N个会议的开始和结束时间,你怎么样安排才能使会议室最大利用?即安排最多场次的会议?
示例:
会议1:0点~9点:9点之前开始的会议都不行了。
会议2:8点~10点
会议3:10点~12点:12点
会议4:8点~20点
结果:最多可以安排两场会议【会议1、会议3】
import java.util.ArrayList; import java.util.List; /** * 题目:某天早上公司领导找你解决一个问题,明天公司有N个同等级的会议需要使用同一个会议室,现在给你这个N个会议的开始和结束时间,你怎么样安排才能使会议室最大利用?即安排最多场次的会议? * * eg: * 会议1:0点~9点:9点之前开始的会议都不行了。 * 会议2:8点~10点 * 会议3:10点~12点:12点 * 会议4:8点~20点 * 结果:最多可以安排两场会议【会议1、会议3】 * * 思路: * 1、将会议根据结束时间进行排序 * 2、遍历会议,如果当前会议开始时间大于等于前一个会议的结束时间,则说明可以安排会议,否则不能安排会议 */ public class MeetingTest { public static void main(String[] args) { List meetings = initInfo(5); meetings.sort(null); int currentTime = 0; for (Meeting meeting : meetings) { if (meeting.getStart() >= currentTime){ System.out.println(meeting); currentTime = meeting.getEnd(); } } } public static List initInfo(int size){ List list = new ArrayList<>(); Meeting meeting1 = new Meeting(0, 9); list.add(meeting1); Meeting meeting2 = new Meeting(8, 10); list.add(meeting2); Meeting meeting3 = new Meeting(10, 12); list.add(meeting3); Meeting meeting4 = new Meeting(8, 20); list.add(meeting4); return list; } } class Meeting implements Comparable{ private int start; private int end; public Meeting(int start, int end) { this.start = start; this.end = end; } public int getStart() { return start; } public int getEnd() { return end; } @Override public int compareTo(Meeting o) { return this.end == o.end ? 0 : this.end < o.end ? -1 : 1; } @Override public String toString() { return "Meeting{" + "start=" + start + ", end=" + end + '}'; } }
动态规划(Dynamic Programming,简称DP)是一种在数学、计算机科学和经济学中使用的,通过把原问题分解为相对简单的子问题的方式来求解复杂问题的方法。动态规划常常适用于有重叠子问题和最优子结构性质的问题。
基本思想
动态规划实现步骤
示例说明
题目:背包问题
问题描述
解决方法
动态规划
定义状态:设f[i] [j]表示前i个物品中选取若干件放入容量为j的背包所能获得的最大价值。
状态转移方程
综合两种情况,可以得到状态转移方程:
初始化:f[0] [j] = 0(j从0到W),表示没有物品可选时,背包的价值为0。
求解:从f[1] [0]到f[n] [W]逐步计算,最终f[n] [W]即为所求的最大价值。
优点
局限性
题目
经典问题:背包问题
小偷去某商店盗窃,背有一个背包,容量是50kg,现在有以下物品(物品不能切分,且只有一个),请问小偷应该怎么拿才能得到最大的价值?
示例
物品1 10kg 60元
物品2 20kg 100元
物品3 30kg 120元
结果:
30+20(kg)=120+100=220
解题思路
状态定义:
物品\背包空间(kg) 10 20 30 40 50
物品1 60 60 60 60 60
物品2 60 100 160 160 160
物品3 60 100 160 180 220
得到状态转移公式:res = MAX(WEIGHT(N) + res(N-1, W-WEIGHT(N)), res(N-1,W)) = MAX(WEIGHT(3) + res(3-1, 5-3), res(3-1,3)) = MAX(120 + 100), 160) = 220
/** * 经典问题:背包问题 * 小偷去某商店盗窃,背有一个背包,容量是50kg,现在有以下物品(物品不能切分,且只有一个),请问小偷应该怎么拿才能得到最大的价值? * * 示例 * 物品1 10kg 60元 * 物品2 20kg 100元 * 物品3 30kg 120元 * * 结果: * 40+10(kg)=120+60=180 * * 解题思路: * 物品\背包空间(kg) 10 20 30 40 50 * 物品1 60 60 60 60 60 * 物品2 60 100 160 160 160 * 物品3 60 100 160 180 220 * * 得到解题公式:res = MAX(WEIGHT(N) + res(N-1, W-WEIGHT(N)), res(N-1,W)) = MAX(WEIGHT(3) + res(3-1, 5-3), res(3-1,3)) = MAX(120 + 100), 160) = 220 */ public class KnapsackProblem { public static void main(String[] args) { int[] weight = {10, 30, 20}; // 物品重量 int[] value = {60, 120, 100}; // 物品价值 int size = 50; // 背包容量 int[][] res = new int[weight.length + 1][(size / 10) + 1]; // 创建二维数组,0行0列用来存放初始值 String[][] resStr = new String[weight.length + 1][(size / 10) + 1]; // 用于存放路径 for (int i = 1; i <= weight.length; i++) { for (int j = 1; j <= size / 10; j++) { if (j < weight[i - 1] / 10){ res[i][j] = res[i - 1][j]; resStr[i][j] = resStr[i - 1][j]; }else{ res[i][j] = Math.max(value[i - 1] + res[i - 1][j - (weight[i - 1] / 10)], res[i - 1][j]); resStr[i][j] = res[i - 1][j - (weight[i - 1] / 10)] + value[i - 1] > res[i - 1][j] ? resStr[i - 1][j - (weight[i - 1] / 10)] + " + " + i : resStr[i - 1][j]; } } } // 打印各个阶段的结果 for (int i = 0; i < res.length; i++) { for (int j = 0; j < res[i].length; j++) { System.out.print(res[i][j] + " "); } System.out.println(); } // 打印各个阶段的路径 for (int i = 0; i < resStr.length; i++) { for (int j = 0; j < resStr[i].length; j++) { System.out.print(resStr[i][j] + " | "); } System.out.println(); } System.out.println("背包可以放的最大价值:"+res[weight.length ][size / 10]); } }
题目
中了一个奖可以清空购物车5000元的东西(不能找零),每个东西只能买一件,那么她应该如何选择物品使之中奖的额度能最大利用呢?
示例
物品1 2000元
物品2 3000元
物品3 4000元
结果:物品1 + 物品2 = 5000元
解题思路
使用动态规划,动态规划的思路是:
values = {2000, 3000, 4000}
物品\价值 1000 2000 3000 4000 5000
1 0 2000 2000 2000 2000
2 0 2000 3000 3000 5000
3 0 2000 3000 4000 5000
得到公式:res[i][j] = max(values[i] + res[i] [V - i], res[i-1] [j])
/** * 题目:中了一个奖可以清空购物车5000元的东西(不能找零),每个东西只能买一件,那么她应该如何选择物品使之中奖的额度能最大利用呢? * * 示例: * 物品1 2000元 * 物品2 3000元 * 物品3 4000元 * * 结果:物品1 + 物品2 = 5000元 * * 思路: * 使用动态规划,动态规划的思路是: * values = {2000, 3000, 4000} * 物品\价值 1000 2000 3000 4000 5000 * 1 0 2000 2000 2000 2000 * 2 0 2000 3000 3000 5000 * 3 0 2000 3000 4000 5000 * * 得到公式:res[i][j] = max(values[i] + res[i][V - i], res[i-1][j]) */ public class ShoppingCartProblem { public static void main(String[] args) { int v = 5000; // 优惠卷的价值 int[] values = {2000, 3000, 4000}; // 物品的价值 int[][] res = new int[values.length + 1][(v / 1000) + 1]; // 记录每一步获取到的最大价值结果 String[][] resStr = new String[values.length + 1][(v / 1000) + 1]; // 记录每一步对应的路径 for (int i = 0; i < values.length; i++) { for (int j = 1; j <= (v / 1000) ; j++) { // 遍历每一个价值 if (j*1000 < values[i]){ res[i + 1][j] = res[i][j]; resStr[i + 1][j] = resStr[i][j]; }else{ res[i + 1][j] = Math.max(values[i] + res[i][j - (values[i] / 1000)], res[i][j]); if (res[i + 1][j] == values[i] + res[i][j - (values[i] / 1000)]){ resStr[i + 1][j] = resStr[i][j - (values[i] / 1000)] + " + " + values[i]; }else{ resStr[i + 1][j] = resStr[i][j]; } } } } // 打印每一个阶段的结果 System.out.println("每一步的背包价值:"); for (int i = 0; i < res.length; i++) { for (int j = 0; j < res[0].length; j++) { System.out.print(res[i][j] + " | "); } System.out.println(); } // 打印每一个阶段的路径 System.out.println("每一步的背包路径:"); for (int i = 0; i < resStr.length - 1; i++) { for (int j = 0; j < resStr[0].length; j++) { System.out.print(resStr[i][j] + " | "); } System.out.println(); } // 得到的最大价值 System.out.println("背包可以放的最大价值:"+res[values.length ][v / 1000]); } }