贪心算法题目总结

news2025/1/21 3:00:32

1. 整数替换

看到这道题目,我们首先能想到的方法就应该是递归解法,我们来画个图

此时我们出现了重复的子问题,就可以使用递归,只要我们遇到偶数,直接将n除以2递归下去,如果是奇数,选出加1和减1中最小步数的那个继续递归下去即可,直接看代码:

class Solution {
public:
    int integerReplacement(int n) {
        if(n == 1)
            return 0;
        if(n & 1 == 1) // 奇数
            return min(integerReplacement(n+1),integerReplacement(n-1)) + 1;
        else
            return integerReplacement(n / 2) + 1;
    }
};

然后我们去运行代码,很遗憾,此时代码没有通过,这是因为的递归过程中出现了大量的重复子问题的,这些重复的子问题其实我们已经计算过了,没必要再计算一次,所以我们可以利用记忆化搜索的备忘录来解决这个问题,此时我们递归的时候,如果此时备忘录立马存在这个值,我们就可以直接返回,如果没有这个值,我们再去递归。

class Solution {
    unordered_map<int,int> hash;
public:
    int integerReplacement(int n) {
        // 先判断是否再在备忘录中
        if(hash.count(n)) 
        {
            return hash[n];
        } 
        if(n == 1)
        {
            hash[1] = 0;
            return 0;
        }  
        if(n & 1 == 1) // 奇数
        {
            hash[n] = min(integerReplacement(n+1),integerReplacement(n-1)) + 1;
            return hash[n];
        }
        else
        {
            hash[n] = integerReplacement(n / 2) + 1;
            return hash[n];
        }  
    }
};

但是此时我们发现程序没有通过,通过这个测试案例我们可以知道,当n是2147483647的时候,此时是奇数,会去递归2147483647+1,此时就超过了int的范围。

此时我们需要自己来定义一个dfs函数来解决这个问题。

class Solution {
    unordered_map<int,int> hash;
public:
    int integerReplacement(int n) 
    {
        return dfs(n);
    }
    int dfs(long long int n){
        // 先判断是否再在备忘录中
        if(hash.count(n)) 
        {
            return hash[n];
        } 
        if(n == 1)
        {
            hash[1] = 0;
            return 0;
        }  
        if(n & 1 == 1) // 奇数
        {
            hash[n] = min(dfs(n+1),dfs(n-1)) + 1;
            return hash[n];
        }
        else
        {
            hash[n] = dfs(n / 2) + 1;
            return hash[n];
        }  
    }
};

此时我们的代码就能通过,我们再来写另一种解法。

直接上手代码:

class Solution {
public:
    int integerReplacement(int n) {
        return dfs(n);
    }
    int dfs(long long int n)
    {
        if(n == 1)
            return 0;
        if(n % 2 == 0) // 偶数
        {
            return dfs(n/2) + 1;
        } 
        else
        {
            if(n == 3)
            {
                return dfs(n-1) + 1;
            }
            else if(n % 4 == 1)
            {
                return dfs(n - 1) + 1;
            }
            else if(n % 4 == 3)
            {
                return dfs(n + 1) + 1;
            }
        }
        // 这里有返回值是因为要让所有的路径都有返回值
        return 0;
    }
};

其实我们这里也可以不使用递归来解决这个题目。

class Solution {
public:
    int integerReplacement(int n) {
        int ret = 0;
        while(n > 1)
        {
            if(n % 2 == 0) // 偶数
            {
                n /= 2;
                ret++;
            }
            else // 奇数
            {
                if(n == 3)
                {
                    n = 1;
                    ret += 2;
                }
                else if(n % 4 == 1)
                {
                    n -= 1;
                    n /= 2;
                    ret += 2;
                }
                else if(n % 4 == 3)
                {
                    n /= 2; // 这里需要先除2,防止n+1溢出
                    n += 1;
                    ret += 2;
                }

            }
        }
    return ret;
    }
};

2. 俄罗斯套娃信封问题

首先我们看到这个题目,就和我们之前的最长递增子序列是一样的,我们需要对左端点进行排序,然后看是否满足俄罗斯套娃的条件,既然是找最长的套娃个数,此时我们可以使用动态规划来解决这个问题,我们可以假设dp[i]位置表示i位置之前的此时最长的套娃个数,当我们到i的位置的时候,我们需要看看0到i-1位置的最长的套娃个数,如果有一个条件满足俄罗斯套娃的条件,那么最长的套娃个数就可以+1,此时就可以使用动态规划来解决这个问题。

class Solution {
public:
    int maxEnvelopes(vector<vector<int>>& envelopes) {
        // 动态规划
        sort(envelopes.begin(), envelopes.end());
        int n = envelopes.size();
        vector<int> dp(n, 1);
        int ret = 1;

        for(int i = 1; i < n; i++)
        {   
            for(int j = 0; j < i; j++)
            {
                if(envelopes[i][0] > envelopes[j][0] &&
                    envelopes[i][1] > envelopes[j][1])
                    {
                        dp[i] = max(dp[i], dp[j]) + 1;
                    }
            }
            ret = max(ret, dp[i]);
        }
        return ret;
    }
};

但是此时我们的代码会超时,所以此时我们就需要来使用另一种策略

class Solution {
public:
    int maxEnvelopes(vector<vector<int>>& envelopes) {
        // 排序
        sort(envelopes.begin(), envelopes.end(), [&](const vector<int>& v1, const vector<int>& v2)
        {
            // 左端点不同的时候
            return v1[0] != v2[0] ? v1[0] < v2[0] : v1[1] > v2[1];
        });

        vector<int> ret;
        ret.push_back(envelopes[0][1]);

        for(int i = 1; i < envelopes.size(); i++)
        {
            int b = envelopes[i][1];
            if(b > ret.back())
            {
                ret.push_back(b);
            }
            else
            {
                int left = 0, right = ret.size() - 1;
                while(left < right)
                {
                    int mid = (left + right) / 2;
                    if(ret[mid] >= b) right = mid;
                    else left = mid + 1;
                }
                ret[left] = b;
            }
        }
        return ret.size(); 
    }
};

3. 可被三整除的最大和

正难则反: 我们可以先把所有的数累加在⼀起,然后根据累加和的结果,贪心的删除⼀些数。

那么我们该如何寻找到最小的和次小的值呢?当然我们可以进行一下sort排序取出最小的和次小的值,但是我们知道sort的底层使用的是快速排序,时间复杂度是O(N*logN),所以此时并不是最优的选择,我们可以使用O(N)的时间复杂度来解决这个问题:

class Solution {
public:
    int maxSumDivThree(vector<int>& nums) {
        int x1 = 0x3f3f3f3f3f;
        int x2 = 0x3f3f3f3f3f;
        int y1 = 0x3f3f3f3f3f;
        int y2 = 0x3f3f3f3f3f;
        int sum = 0;
        // 求出总和,最小和次小的值
        for(int i = 0; i < nums.size(); i++)
        {
            sum += nums[i];
            if(nums[i] % 3 == 1) // 判断当值余数是否为1
            {
                if(nums[i] <= x1)
                    x2 = x1, x1 = nums[i];
                else if(nums[i] > x1 && nums[i] < x2)
                    x2 = nums[i];
            }
            else if(nums[i] % 3 == 2) // 判断当值余数是否为2
            {
                if(nums[i] <= y1)
                    y2 = y1, y1 = nums[i];
                else if(nums[i] > y1 && nums[i] < y2)
                    y2 = nums[i];
            }
        }
        // 分类讨论
        if(sum % 3 == 0) return sum;
        else if(sum % 3 == 1) return max(sum - x1, sum - y1 - y2);
        else return max(sum - y1, sum - x1 - x2);
    }
};

4. 距离相等的条形码

这道题目比较简单,我们只需要每次处理一批相同的数字,往 n 个空里面摆放; 每次摆放的时候,隔一个格子摆放一个数,但是我们需要优先处理出现次数最多的那个数,其余的数处理的顺序就没什么要求了。

class Solution {
public:
    vector<int> rearrangeBarcodes(vector<int>& barcodes) {
        unordered_map<int, int> hash; // 统计每个数出现的次数
        int maxVal = 0; // 最大的值
        int maxCount = 0; // 最大值的项数
        for(auto& e : barcodes)
        {
            if(maxCount < ++hash[e])
            {
                maxCount = hash[e];
                maxVal = e;
            }
        }
        vector<int> v(barcodes.size(), 0);
        // 先处理出现次数最多的数
        int index = 0;
        for(int i = 0; i < maxCount; i++)
        {
            v[index] = maxVal;
            index += 2;
        }
        // 处理剩下的数,不用管顺序,直接防即可
        hash.erase(maxVal);
        for(auto& [x, y] : hash)
        {   
            for(int i = 0; i < y; i++)
            {
                // 超过最大位置,需要重新从头开始摆放
                if(index >= barcodes.size()) 
                    index = 1;
                v[index] = x;
                index += 2;
            }
        }
        return v;
    }
};

5. 重构字符串

我们读完题目会发现这道题目和上一道题目是类似的,只不过上一道题目是重排数字,而这道题目是重排字符串,思路基本上都是一致的,但是上一个题目明确告诉了我们所有数字都是可以重排列的,但是我们这个题目不一定,因此我们需要先判断一下,如果出现最多的字符的个数大于(n + 1) / 2 就无法重排,其他情况下就可以直接重排列,那我们直接上代码啦!

class Solution {
public:
    string reorganizeString(string s) {
        unordered_map<char, int> hash;
        char maxVal = ' ';
        int maxCount = 0;
        for(auto& x : s)
        {
            if(maxCount < ++hash[x])
            {
                maxCount = hash[x];
                maxVal = x;
            }
        }

        if(maxCount > ((s.size() + 1) / 2))
            return "";
        // 先处理出现次数最多的那个字符
        string ret;
        ret.resize(s.size());
        int index = 0;
        for(int i = 0; i < maxCount; i++)
        {
            ret[index] = maxVal;
            index += 2;
        }
        hash.erase(maxVal); // 删除最多元素项的值
        // 处理剩余的值
        for(auto& [x, y] : hash)
        {
            for(int i = 0; i < y; i++)
            {
                if(index >= ret.size()) index = 1;
                ret[index] = x;
                index += 2;
            }
        }
        return ret;

    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL 高级SQL高级语句(二)

一.CREATE VIEW 视图 可以被当作是虚拟表或存储查询。 视图跟表格的不同是&#xff0c;表格中有实际储存数据记录&#xff0c;而视图是建立在表格之上的一个架构&#xff0c;它本身并不实际储存数据记录。 临时表在用户退出或同数据库的连接断开后就自动消失了&#xff0c;而…

模型预测控制:设定点跟踪(Set Point Tracking)

模型预测控制&#xff1a;设定点跟踪&#xff08;Set Point Tracking&#xff09; 模型预测控制&#xff08;Model Predictive Control, MPC&#xff09;不仅可以用于系统稳定性问题&#xff0c;还可以用于设定点跟踪问题&#xff08;Set Point Tracking&#xff09;&#xff…

13_旷视轻量化网络--ShuffleNet V2

回顾一下ShuffleNetV1:08_旷视轻量化网络--ShuffleNet V1-CSDN博客 1.1 简介 ShuffleNet V2是在2018年由旷视科技的研究团队提出的一种深度学习模型&#xff0c;主要用于图像分类和目标检测等计算机视觉任务。它是ShuffleNet V1的后续版本&#xff0c;重点在于提供更高效的模…

架构师篇-10、DDD实战篇:通过领域模型落地系统

基于领域模型的设计与开发 数据库设计程序设计微服务设计 在线订餐系统的领域事件通知 微服务拆分 事件风暴会议 梳理领域事件进行领域建模识别聚合关系划分限界上下文 用户下单领域模型 更新后的模型 领域模型的设计实现过程 数据库设计 数据库映射&#xff1a;一对一关系…

cesium 添加 Echarts 图层(空气质量点图)

cesium 添加 Echarts 图层(下面附有源码) 1、实现思路 1、在scene上面新增一个canvas画布 2、通坐标转换,将经纬度坐标转为屏幕坐标来实现 3、将ecarts 中每个series数组中元素都加 coordinateSystem: ‘cesiumEcharts’ 2、示例代码 <!DOCTYPE html> <html lan…

EDA期末复习——基础知识

个人名片&#xff1a; &#x1f393;作者简介&#xff1a;嵌入式领域优质创作者&#x1f310;个人主页&#xff1a;妄北y &#x1f4de;个人QQ&#xff1a;2061314755 &#x1f48c;个人邮箱&#xff1a;[mailto:2061314755qq.com] &#x1f4f1;个人微信&#xff1a;Vir2025WB…

iconfont-阿里巴巴矢量图标库 在vue项目使用记录

官网地址&#xff1a;https://www.iconfont.cn/manage/index?manage_typemyprojects&projectId4539761 第一步&#xff1a; 下载资源 ->解压到项目文件夹 第二步 在项目中main.ts 或者main.js 引入资源 import //assets/iconfont/font/iconfont.js; import //assets…

【考研408计算机组成原理】微程序设计重要考点指令流水线考研真题+考点分析

苏泽 “弃工从研”的路上很孤独&#xff0c;于是我记下了些许笔记相伴&#xff0c;希望能够帮助到大家 目录 微指令的形成方式 微指令的地址形成方式 对应考题 题目&#xff1a;微指令的地址形成方式 - 断定方式 解题思路&#xff1a; 答题&#xff1a; 分析考点&…

【训练篇】MLU370-M8 完成 qwen1.5-7b-chat-lora训练及推理

文章目录 前言一、平台环境配置二、环境 or 模型准备1.模型下载2.环境准备2.1 modelscope2.2 transformers2.3 accelerate2.4 deepspeed2.5 peft2.6 环境代码修改 3训练代码准备4 代码修改 三&#xff0c;训练后推理验证四.推理效果展示1.微调前2.微调后 前言 本期我们采用魔塔…

局域网必备文件传输神器,吾爱再出精品,支持电脑、手机无缝对接!

今天给大家带来的不是一般的干货&#xff0c;而是一款让阿星我爱不释手的局域网文件传输神器&#xff0c;而且是吾爱大佬出品。无论是工作还是生活&#xff0c;它都能给你带来极大的便利。这年头&#xff0c;谁还没个跨设备传输文件的需求呢&#xff1f; 手机、电脑、平板&…

一个中文和越南语双语版本的助贷平台开源源码

一个中文和越南语双语版本的助贷平台开源源码。后台试nodejs。 后台 代理 前端均为vue源码&#xff0c;前端有中文和越南语。 前端ui黄色大气&#xff0c;逻辑操作简单&#xff0c;注册可对接国际短信&#xff0c;可不对接。 用户注册进去填写资料&#xff0c;后台审批&…

python(一)下载安装入门

一.下载安装 1、官网下载地址&#xff1a; Python Releases for Windows | Python.org 2、下载安装 1.下载python包&#xff1a;点击下载 2.安装 2.默认点击next即可 3.选择你想安装的路径&#xff0c;点击install即可 4.这里如果出现管理员字样&#xff0c;点击授权即可 安…

专题一: Spring生态初探

咱们先从整体脉络上看下Spring有哪些模块&#xff0c;重要的概念有个直观印象。 从Spring框架的整体架构和组成对整体框架有个认知。 Spring框架基础概念 Spring基础 - Spring和Spring框架组成 上图是从官网4.2.x获取的原图&#xff0c;目前我们使用最广法的版本应该都是5.x&am…

医院管理系统带万字文档医院预约挂号管理系统基于spingboot和vue的前后端分离java项目java课程设计java毕业设计

文章目录 仓库管理系统一、项目演示二、项目介绍三、万字项目文档四、部分功能截图五、部分代码展示六、底部获取项目源码带万字文档&#xff08;9.9&#xffe5;带走&#xff09; 仓库管理系统 一、项目演示 医院管理系统 二、项目介绍 基于springbootvue的前后端分离医院管…

鱼叉式钓鱼

鱼叉式网络钓鱼&#xff1a; 鱼叉式网络钓鱼是一种网络钓鱼形式&#xff0c;它针对特定个人或组织发送定制消息&#xff0c;旨在引发特定反应&#xff0c;例如泄露敏感信息或安装恶意软件。这些攻击高度个性化&#xff0c;使用从各种来源收集的信息&#xff0c;例如社交媒体资…

运维锅总详解Prometheus

本文尝试从Prometheus简介、架构、各重要组件详解、relable_configs最佳实践、性能能优化及常见高可用解决方案等方面对Prometheus进行详细阐述。希望对您有所帮助&#xff01; 一、Prometheus简介 Prometheus 是一个开源的系统监控和报警工具&#xff0c;最初由 SoundCloud …

【简易版tinySTL】 红黑树- 定义, 插入, 构建

文章目录 旋转左旋右旋 左旋右旋代码实现红黑树的基本性质红黑树的插入红黑树的插入示例红黑树修复代码实现参考资料 旋转 对于一个平衡二叉搜索树&#xff0c;左子树高度为4&#xff0c;右子树高度为2&#xff0c;它们的高度差为2&#xff0c;破坏了平衡性&#xff08;高度差&…

扩展阅读:什么是中断

如果用一句话概括操作系统的原理,那就是:整个操作系统就是一个中断驱动的死循环,用最简单的代码解释如下: while(true){doNothing(); } 其他所有事情都是由操作系统提前注册的中断机制和其对应的中断处理函数完成的。我们点击一下鼠标,敲击一下键盘,执行一个程序,…

忙忙碌碌的混沌之中差点扑了个空而错过年中这条线

文章目录 前言初见端倪混沌初始力不从心心力交瘁拾遗补缺总结 前言 突然意识到过完这个周末已经7月份了&#xff0c;他预示着我的2024年已经过半了&#xff0c;过年回家仿佛还是昨天的事情&#xff0c;怎么转眼间已经到了年中了。心里还是不愿承认这件事&#xff0c;翻开自己2…

使用NFS网关功能将HDFS挂载到本地系统

HDFS安装教程 HDFS安装教程http://t.csdnimg.cn/2ziFd 使用NFS网关功能将HDFS挂载到本地系统 简介 HDFS提供了基于NFS&#xff08;Network File System&#xff09;的插件&#xff0c;可以对外提供NFS网关&#xff0c;供其它系统挂载使用。 NFS 网关支持 NFSv3&#xff0c;并…