贪心算法

贪心算法

贪心算法:指再对问题求解时,总是做出在当前看来是最好的选择,就是指不同全局考虑问题,做到了局部的最优解。

看了上面的定义,我们在实际应用中,也会懵逼,何时可以用贪心算法?
我们可以从主管角度理解为,贪心算法是问题的解可以通过一步步最优解得到的,总之,贪心算法类问题就那么几类,我们通过这几类问题去了解贪心算法吧

1.背包问题

问题:假设我们有一个容量为100kg的包,现在有一下物品,我们要求在有限容量的条件下,让背包的总价值最大。


tx1.jpg

贪心,就是我们要总价值最大,我们不必先考虑全局,我们就尽可能将价值比最大的物品多装入背包,这个思想就是贪心了。
黄豆的性价比 = 1
绿豆的性价比 = 3
红豆的性价比 = 2
黑豆的性价比 = 4
青豆的性价比 = 1.5
那我们肯定先装黑豆,全部装入,背包剩余80kg,接下来装绿豆,全部装入背包剩余50kg,接下来只能装50kg红豆了。
选择方案:
20kg黑豆+30kg绿豆+50kg红豆 总价值 = 270

void back_Pack()
{
    double sumW = 100, sumV = 0;//背包最大容量100kg
    double weight[] = { 100,30,60,20,50 };
    double value[] = { 100,90,120,80,75 };
    int temp[5] = { 0 }, index;//标记当前物品选择与否
    double bi[5] = { 0 }, maxBi;//性价比
    for (int i = 0; i < 5; i++)
    {
        bi[i] = value[i] * 1.0 / weight[i];
    }
    //开始挑选物品
    for (int i = 0; i < 5; i++)
    {
        maxBi = -9999;
        //挑选当前未选的物品性价比最大的
        for (int j = 0; j < 5; j++)
        {
            if (temp[j] == 0 && bi[j] > maxBi)
            {
                maxBi = bi[j];
                index = j;
            }
        }
        //背包容量不足
        if (sumW <= 0)
        {
            break;
        }
        //背包容量足
        temp[index] = 1;
        if (sumW >= weight[index])
        {
            sumW -= weight[index];//全部放入
            sumV += value[index];
        }
        else {
            sumV += bi[index] * sumW;
            sumW = 0;//容量不足,放入一部分
        }
    }
    //输出最大价值
    printf_s("%lf", sumV);
}

2.活动安排问题

问题:假设现在有几个活动,每个活动用区间[a,b]表示,区间表示活动在时间a开始,时间b结束,且在任何时刻只能进行一个活动,即活动1在某个时间段进行时,后面但凡有活动的区间在活动1范围内,这个活动是不可进行的。
比如,活动1[3,6],活动2[1,4],活动3[6,7]活动1和活动2是不可以同时进行的,活动1和活动3可以同时进行。
题目问题是,在给的个活动中,求最大可以安排的活动数目。

活动编号 开始时间 结束时间
1 1 3
2 2 5
3 4 8
4 6 10

分析问题:我们如何让活动安排更多?,采用贪心的策略就是我么每次选择一个可以给后面剩余更多时间的安排,所以,将所有活动按开始时间递增排序。
上面表就是按照活动开始时间递增排序的,首先选择活动1,活动2不选,和活动1冲突了,活动3可以选,活动4不可以选,和活动3冲突。
根据贪心法,我们选择活动1和活动3,最大安排个数是2个。

 void plan_Acivity()
{
    struct plan s[11] = { { 2,5 },{ 4,8 },{ 6,10 },{ 1,3 } };
    struct plan temp;
    int sum = 0;
    //按照活动开始时间排序
    for (int i = 0; i < 4; i++)
    {
        for (int j = 0; j < 4 - i - 1; j++)
        {
            if (s[j].begin > s[j + 1].begin)
            {
                temp = s[j];
                s[j] = s[j + 1];
                s[j + 1] = temp;
            }
        }
    }
    //开始逐个安排活动
    temp = s[0];
    sum++;
    for (int i = 1; i < 4; i++)
    {
        if (s[i].begin > temp.last)//判断次活动是否可以安排
        {
            sum++;
            temp = s[i];
        }

    }
    printf_s("%d", sum);
}

3.最优装载问题

问题:在一艘轮船上,容量最大为w,现在有一批货物,它们重量分别是W(i),为了不超过轮船最大装载,我们要装尽可能多的货物,如何装载?

物品编号 物品重量
1 5
2 2
3 5
4 4
5 3

这个问题就十分符合我们主观上的贪心策略,我们首先将所有货物按照重量从小到大排序,然后从第一个后取,直到装不下为止。

所以选择编号为 2,5,4货物去装载。

void load()
{
    int weight[5] = { 5,2,6,4,3 }, temp, maxW = 10;
    for (int i = 0; i < 5; i++)
    {
        for (int j = 0; j < 5 - i - 1; j++)
        {
            if (weight[j] > weight[j + 1])
            {
                temp = weight[j];
                weight[j] = weight[j + 1];
                weight[j + 1] = temp;
            }
        }
    }
    //开始逐个装载
    for (int i = 0; i < 5; i++)
    {
        if (maxW - weight[i] > 0)
        {
            printf_s("%d\n", weight[i]);
            maxW -= weight[i];
        }
    }
}

4.多机调度问题

设有n个独立的作业,由m台独立的机器进行这些作业的处理,每个作业需要处理的时间是不同的,每个作业均可在任何一个机器上进行,多机调度问题要求给出一种方案,使得所有任务在最短时间内在m台机器上完成作业。
假设机器个数为3个任务是7个任务,下表是每个任务执行时间

任务编号 处理时间
1 2
2 14
3 4
4 16
5 6
6 5
7 3

我们的策略上,用贪心思想,我们要最短时间完成所有的任务,那我们先安排最长时间的任务,让最长任务进行的同时,其它任务在同步进行。将任务按照处理时间递减排序。如下表

任务编号 处理时间
4 16
2 14
5 6
6 5
3 4
7 3
1 2

机器1分配任务4,执行时间16,占用时间段[0,16]
机器2分配任务2,执行时间14,占用时间段[0,14]
机器3分配任务5,执行时间6,占用时间段[0,6]
机器3分配任务6,执行时间5,占用时间段[6,11]
机器3分配任务3,执行时间4,占用时间段[11,15]
机器2分配任务7,执行时间3,占用时间段[14,17]
机器3分配任务1,执行时间2,占用时间段[15,17]
所以最短可在17小时内完成。

5.哈夫曼编码问题

这个问题我们已经在大一离散,大二的数据结构,算法,都直到了,我就不写详细过程了,只分析它是贪心思想,每次找出权值最小的两个点合并。。。。

获取完整代码

我分别用C,C++,JAVA三种主流语言编写了完整代码,请大家指导批正,一起学习。

点击查看

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容

  • 概述 贪心算法(英语:greedy algorithm),又称贪婪算法,是一种在每一步选择中都采取在当前状态下最好...
    jiaji_3740阅读 3,865评论 0 1
  • 可用贪心算法解决的几个基本问题 关键:看问题有没有贪心选择性质和最优子结构性质。有些问题看似是可以用贪心算法,但是...
    碧影江白阅读 6,208评论 1 2
  • 前言 贪心是人类自带的能力,贪心算法是在贪心决策上进行统筹规划的统称。 比如一道常见的算法笔试题----跳一跳: ...
    落影loyinglin阅读 41,776评论 11 84
  • 一、双赢思维。自己的成功不一定要建立在他人的失败之上,共同进步,利人利己才是长久之道。 二、知彼解己。想让他人理解...
    圆舞amour阅读 174评论 1 3
  • 函数 作用:复用代码(可复用多行代码) 函数的定义: 函数的调用:函数名加一个括号表示调用 函数名();例如:fn...
    Metallic阅读 420评论 0 0