经典算法之贪心算法

Posted 爱编码

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了经典算法之贪心算法相关的知识,希望对你有一定的参考价值。

什么是贪心算法?

  贪心算法,又称贪婪算法(Greedy Algorithm),是指在对问题求解时,总是做出在当前看来是最好的选择。也就是说,不从整体最优解出发来考虑,它所做出的仅是在某种意义上的局部最优解。  

贪婪算法是一种分阶段的工作,在每一个阶段,可以认为所做决定是最好的,而不考虑将来的后果。这种“眼下能够拿到的就拿”的策略是这类算法名称的来源。  

贪心算法没有固定的算法框架,算法设计的关键是贪心策略的选择。必须注意的是,贪心算法不是对所有问题都能得到整体最优解,选择的贪心策略必须具备无后效性,即某个状态以后的过程不会影响以前的状态,只与当前状态有关。所以对所采用的贪心策略一定要仔细分析其是否满足无后效性。

贪心算法的基本思路:

  1. 建立数学模型来描述问题。

  2. 把求解的问题分成若干个子问题。

  3. 对每一子问题求解,得到子问题的局部最优解。

  4. 把子问题的解局部最优解合成原来解问题的一个解。

贪心算法适用的问题

  贪心策略适用的前提是:局部最优策略能导致产生全局最优解。也就是当算法终止的时候,局部最优等于全局最优。

贪心算法的实现框架

从问题的某一初始解出发;

while (能朝给定总目标前进一步) { 利用可行的决策,求出可行解的一个解元素;

} 由所有解元素组合成问题的一个可行解;

贪心策略的选择

  因为用贪心算法只能通过解局部最优解的策略来达到全局最优解,因此,一定要注意判断问题是否适合采用贪心算法策略,找到的解是否一定是问题的最优解。如果确定可以使用贪心算法,那一定要选择合适的贪心策略;

贪心算法的几个例子

1. 纸币找零问题

假设1元、2元、5元、10元、20元、50元、100元的纸币,张数不限制,现在要用来支付K元,至少要多少张纸币?

很显然,我们很容易就想到使用贪心算法来解决,并且我们所根据的贪心策略是,每一步尽可能用面值大的纸币即可。当然这是正确的,代码如下:

 
   
   
 
  1. /**

  2. * 钱币找零问题

  3. *

  4. * @param money the money

  5. */

  6. public static void greedyGiveMoney(int money) {

  7. System.out.println("需要找零: " + money);

  8. int[] moneyLevel = {1, 5, 10, 20, 50, 100};

  9. for (int i = moneyLevel.length - 1; i >= 0; i--) {

  10. int num = money/ moneyLevel[i];

  11. int mod = money % moneyLevel[i];

  12. money = mod;

  13. if (num > 0) {

  14. System.out.println("需要" + num + "张" + moneyLevel[i] + "块的");

  15. }

  16. }

  17. }

(1)如果不限制纸币的金额,那这种情况还适合用贪心算法么。比如1元,2元,3元,4元,8元,15元的纸币,用来支付K元,至少多少张纸币?

经我们分析,这种情况是不适合用贪心算法的,因为我们上面提供的贪心策略不是最优解。比如,纸币1元,5元,6元,要支付10元的话,按照上面的算法,至少需要1张6元的,4张1元的,而实际上最优的应该是2张5元的。

(2)如果限制纸币的张数,那这种情况还适合用贪心算法么。比如1元10张,2元20张,5元1张,用来支付K元,至少多少张纸币?

同样,仔细想一下,就知道这种情况也是不适合用贪心算法的。比如1元10张,20元5张,50元1张,那用来支付60元,按照上面的算法,至少需要1张50元,10张1元,而实际上使用3张20元的即可;

(3)所以贪心算法是一种在某种范围内,局部最优的算法。

2.股票问题

题干条件:假设你有一个数组,其中第i个元素是某只股票在第i天的价格。如你最多只获准完成一项交易(即,买一股,卖一股),设计一个算法来寻找最大的利润。注意,你不能在买股票之前卖掉它。

测试样例:

 
   
   
 
  1. Example 1:


  2. Input: [7,1,5,3,6,4]

  3. Output: 5

  4. Explanation: Buy on day 2 (price = 1) and sell on day 5 (price = 6), profit = 6-1 = 5.

  5. Not 7-1 = 6, as selling price needs to be larger than buying price.

  6. Example 2:


  7. Input: [7,6,4,3,1]

  8. Output: 0

  9. Explanation: In this case, no transaction is done, i.e. max profit = 0.

Java代码实现:

 
   
   
 
  1. public static int maxProfit(int[] prices) {

  2. //1.很巧妙的思路 一层循环解决 更新数组里面的最小值 和最大插值

  3. Integer maxProfit = Integer.MIN_VALUE;

  4. Integer minPrice = Integer.MAX_VALUE;

  5. for(Integer price:prices){

  6. minPrice=Math.min(minPrice,price);

  7. maxProfit = Math.max(maxProfit,price-minPrice);

  8. }

  9. return maxProfit<=0?0:maxProfit;

  10. }

参考文档:

https://www.cnblogs.com/xiaozhang2014/p/7783795.html

http://www.pianshen.com/article/5371279467/

推荐阅读


2、

3、


以上是关于经典算法之贪心算法的主要内容,如果未能解决你的问题,请参考以下文章

48.贪心算法之四:跳跃游戏问题

java贪心算法经典案例之项目利润最大问题

47.贪心算法之三:如果可以预知未来,股票最佳买卖时机问题

49.贪心算法之五:花园浇水问题

那些经典算法:贪心算法

经典算法题 :贪心算法(大众点评笔试题)