贪心算法篇

2024-02-29 1796阅读

温馨提示:这篇文章已超过395天没有更新,请注意相关的内容是否还可用!

贪心算法篇

“靠漫步,将生趣填饱~” 


贪心算法简介?

         贪心算法(Greedy Algorithm),也称为贪婪算法,是一种在解决问题时采取贪心策略的方法。其基本原理是很简单的:

        “在每个决策点上都选择当下看似最好的选项,而不是寻求全局最优点”。

        我们举几个,常使用贪心算法的小例:

找零问题:

        此时你的顾客一手掏出50“大米”递给你,一手拿着一瓶快乐水——“nutrition happy line”(you know,这瓶饮料的价格为4)。现如今,这位顾客正一脸疑惑地盯着你的一举一动,因为你接过纸币后,目不转睛地瞅着那数字不小的“大米”愣神。你总会在感到一股苍劲的凉风过后,两眼冒星,腥咸的液体会被你从口中送入食管——你应当马上给他找零了。拉开你正下方,散发着浓烈朽木味儿的抽屉,你从中看到了无数的纸币,其中的面额如下:[20,  10 , 5 ,1]。你需要使用最少的纸币,完成找零工作:

        已知,我们要给这位虎背熊腰的壮汉的找零数是:46。又要求我们使用最小的纸币数,所以,我们将两张黄旧的、纸面油印为20的纸币重叠好,再选取面额分别为5和1的纸币,一并夹在手指之间,塞给了这位壮汉。我们的选择为:20 * 2 + 5 * 1 + 1 * 1 = 46。总共需要四张纸币,完成这份找零工作。这便是最少使用纸币的解法。

最小路径和:

        这天你命犯桃花,因为本应对你爱答不理、而你却日夜心念的邻家小妹邀请你同她加入到这一场由神秘人创办的乐园探险中。你本以为这仅仅只是一场普通的游乐主题,彼时暗自窃喜,怀揣着想入非非的心思,幻想着邻家小妹把你相拥、同你腻歪的恋爱场景。然而,这场游戏完完全全没有表面看起来那么简单,处处透露着诡异、怪诞,你莫名被卷入到了一场恐怖的布局和惊天的阴谋之中,感受来自黑暗的惊悚:消失的人脸、怪异的乞丐、脱落的车轨以及血腥、压抑的迷宫……

        每个格子的数字,代表着探寻这个九宫格格子的时间。你需要花最少的时间,进入到右下角的最后一个格子之中,从恶魔的祭奠仪式拯救邻家小妹……

贪心算法篇

        上述的两个例子,对于第一个例子而言,选择的方案:“尽可能选择较大面额的纸币” ,最终我们可以得到“最优解”。相反,对于第二个例子而言,我们的选择是 “选择花费时间较少的格子”进行探索,然而事实上,得出的并不是最优解。

        贪心算法通常会逐步构建问题的解空间,每次尝试将下一个待选元素加入到解集中,直到无法再添加为止。这个过程会使得问题简化为一系列子问题,每个子问题都可以通过同样的贪婪策略来解决,从而逐步接近整体的最优解。

        所谓的这些从局部的角度考虑选择的方案,实质上就是“贪心”策略。然而,“贪心”策略也可能是“错误”的方法,让我们得不出最有解。所以,正确的“贪心”策略,是需要进行验证、证明的。


柠檬水找零    

(1) 题目解析

贪心算法篇

(2) 算法原理              

贪心算法篇

class Solution {
public:
    bool lemonadeChange(vector& bills) {
        // 记录5$ 10$的个数
        int five = 0,ten = 0;
        for(auto& bill:bills)
        {
            if(bill == 5) five++; // 5$ 直接收下
            else if(bill == 10)
            {
                if(five == 0) return false;  // 没有5$ 不能找零
                else five--;
                ten++;  // 收下10$
            }
            else
            {
                if(five && ten) five--,ten--; // 贪心策略:尽量保留5$
                else if(five > 2) five -= 3;
                else return false;
            }
        }
        return true;
    }
};

 贪心证明:

        贪心只是一种策略,考虑的角度也仅仅是局部的“最优解”,所以,贪心策略也可能是“错误的”

如何确定贪心求得的解就是最优解,还需要进行证明求真。

🥃 证明策略1:交换论证

贪心算法篇

贪心算法篇


将数组和减半的最少操作次数         

(1) 题目解析        贪心算法篇

(2) 算法原理

贪心算法篇

class Solution {
public:
    int halveArray(vector& nums) {
        priority_queue pq;
        double sum = 0;
        for(auto n:nums)
        {
            pq.push(n);
            sum += n;  
        }
        sum /= 2.0;
        // 数组减半
        int count = 0; // 记录操作次数
        while(sum > 0)
        {
            double top = pq.top();
            pq.pop();
            top /= 2.0;
            sum -= top;
            pq.push(top);
            count++;
        }
        return count;
    }   
};

贪心证明:

        🍷 交换论证法:

贪心算法篇


最大数

(1) 题目解析

贪心算法篇

(2) 算法原理

贪心算法篇

class Solution {
public:
    string largestNumber(vector& nums) {
        vector strs;
        for(auto x:nums) strs.push_back(to_string(x));
        sort(strs.begin(),strs.end(),[](const string& s1,const string& s2)
        {
            return s1 + s2 > s2 + s1;
        });
        // 提取结果
        string res;
        for(auto& s:strs) res += s;
        // 处理前置0
        if(res[0] == '0') return "0";
        return res;
    }
};

贪心证明:

        似乎,没有看到本题的贪心策略呢? 贪心在何处?

贪心算法篇


摆动序列

(1) 题目解析

贪心算法篇

(2) 算法原理        贪心算法篇

class Solution {
public:
    int wiggleMaxLength(vector& nums) {
        if(nums.size() 
VPS购买请点击我

免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们,邮箱:ciyunidc@ciyunshuju.com。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!

目录[+]