【Py/Java/C++三种语言详解】LeetCode每日一题240122【贪心】LeetCode670、最大交换
文章目录
- 题目链接
- 题目描述
- 解题思路
- 为什么是贪心
- 一个带图的例子
- 代码
- python
- java
- cpp
- 时空复杂度
- 华为OD算法/大厂面试高频题算法练习冲刺训练
题目链接
LeetCode670、最大交换
题目描述
给定一个非负整数数组 nums
和一个整数 k
,你需要将这个数组分成 k
个非空的连续子数组。
设计一个算法使得这 k
个子数组各自和的最大值最小。
示例 1:
输入:nums = [7,2,5,10,8], k = 2
输出:18
解释: 一共有四种方法将 nums 分割为 2 个子数组。 其中最好的方式是将其分为 [7,2,5] 和 [10,8] 。 因为此时这两个子数组各自的和的最大值为18,在所有情况中最小。
示例 2:
输入:nums = [1,2,3,4,5], k = 2
输出:9
示例 3:
输入:nums = [1,4,4], k = 3
输出:4
提示:
1 <= nums.length <= 1000
0 <= nums[i] <= 10(6)
1 <= k <= min(50, nums.length)
解题思路
数据范围(数字的长度)最大为8
,时间复杂度为O(N^3)
的暴力法可以通过。
所谓暴力法,就是枚举出所有不同的下标对(i, j)
,交换s[i]
和s[j]
,找到交换完之后最大的那一组。
思路较为简单,故在此略去不表。一下讨论贪心的做法。
为什么是贪心
由于最多只能交换一次,贪心地思考一下这个问题:我们什么希望进行一个怎么样的交换?
换言之,怎么交换才能使得数字尽可能地大?
考虑例子
9091987
原字符串中的第三个"9"
是最大且位置尽可能靠后的数字,这个字符应该优先地被交换到尽可能前的位置。由于索引0
的数字是"9"
,所以考虑索引1
的字符"0"
和第三个"9"
交换。得到答案
9991087
从这个例子可以看出贪心的策略是:
- 首选一个尽可能大的数字(比如示例中选择字符
"9"
) - 如果有多个最大的数字,则优先选择位置尽可能靠后的那个(比如示例中选择第三个
"9"
) - 将该数字交换到尽可能靠前的位置,即交换到第一个小于该数字的位置(比如示例中索引
1
的位置)。
所以考虑逆序遍历原数字字符串(为了方便交换操作,改成数组来操作),并且使用一个栈(类似一个单调栈),储存原数字从右往左看遇到的更大的数字的下标
stack = list()
for i in range(n-1, -1, -1):if not stack or lst[i] > lst[stack[-1]]:stack.append(i)
最终这个栈一定会满足以下条件:
- 栈中储存的是原数字字符串的数字的下标
i
i
的取值自栈底向栈顶递减,即栈顶元素stack[-1]
是在数字lst
中位置最靠前的下标(满足了上述贪心策略2
)lst[i]
的取值自栈底向栈顶递增,即栈顶元素对应的下标在数字数组中的取值lst[i]
是最大的数字(满足了上述贪心策略1
)
以例子num = 9091987
为例,栈中的结果是储存了最后三个数字"987"
的下标,即stack = [6, 5, 4]
接下来要考虑如何实现上述贪心策略的第三点。
我们可以从头到尾遍历原数字数组lst
,将下标i
和栈顶元素stack[-1]
、以及下标i
对应的数字lst[i]
和栈顶元素对应的数字lst[stack[-1]]
进行比较。若
i < stack[-1]
,说明此时下标i
的位置位于stack[-1]
的左边,可以继续进行后续判断。若lst[i] < lst[stack[-1]]
,说明此时可以交换位置i
和stack[-1]
的两个数字,交换之且退出循环lst[i] >= lst[stack[-1]]
,说明此时不能进行交换,i
需要继续增大
i >= stack[-1]
,说明此时下标i
的位置已经不再位于stack[-1]
的左边,此时不能再考虑栈顶元素,应该将其弹出
另外,由于涉及弹出操作,如果出现空栈情况,但尚未进行交换,则说明原数字数字本身就是一个非递增序列,需要退出循环。综上,上述贪心操作的代码为
for i in range(n):if not stack:breakif i > stack[-1]:if lst[i] < lst[stack[-1]]:lst[i], lst[stack[-1]] = lst[stack[-1]], lst[i]ans = "".join(lst)breakelse:continueelse:stack.pop()
一个带图的例子
再举一个例子,num = "9987687676"
,答案应该为ans = "9988677676"
,可以做出如下图
逆序遍历数组lst
,构建栈stack = [8, 7, 4, 1]
,为可能进行交换的那些对应较大数字且靠后的位置。
正序遍历i
,反复拿出栈顶索引对应的元素lst[stack[-1]]
对应的数字和i
对应的元素lst[i]
进行比较。会经历如下过程。
i < stack[-1]
,但lst[i] >= lst[stack[-1]]
。不能做交换,i
增加。
i >= stack[-1]
,即i
的位置不位于stack[-1]
的左边,stack[-1]
出栈,i
增加。
i < stack[-1]
,但lst[i] >= lst[stack[-1]]
。不能做交换,i
增加。
i < stack[-1]
,且lst[i] < lst[stack[-1]]
。进行交换,得到ans = "9988677676"
,是可以得到的数字最大的结果。
代码
python
# 贪心+栈:O(N)
class Solution:def maximumSwap(self, num: int) -> int:# 用列表的形式储存数字numlst = list(str(num))# 获得数字num的位数(即lst的长度)n = len(lst)# 构建一个栈,储存原字符串从右往左看遇到的更大数字的下标stack = list()# 逆序遍历字符串sfor i in range(n-1, -1, -1):# 如果栈是空栈,或者当前下标i对应的数字lst[i]大于栈顶下标对应的数字lst[stack[-1]]# 则将索引i加入stackif not stack or lst[i] > lst[stack[-1]]:stack.append(i)# 正序遍历列表lstfor i in range(n):# 若出现空栈情况,则退出循环if not stack:break# 如果当前下标i位于栈顶元素stack[-1]的左边# 则可以进行后续判断if i < stack[-1]:# 若当前数字小于栈顶元素对应的数字,则可以进行交换if lst[i] < lst[stack[-1]]:lst[i], lst[stack[-1]] = lst[stack[-1]], lst[i]return int("".join(lst))# 否则,考虑下一个i,这里的else也可以不写else:continue# 如果当前下标i不位于栈顶元素stack[-1]的左边# 则弹出栈顶元素,考虑下一个较小但是位于较右位置的数字else:stack.pop()return num
java
public class Solution {public int maximumSwap(int num) {char[] chars = Integer.toString(num).toCharArray();int n = chars.length;int[] stack = new int[n];int top = -1;for (int i = n - 1; i >= 0; i--) {if (top == -1 || chars[i] > chars[stack[top]]) {stack[++top] = i;}}for (int i = 0; i < n; i++) {if (top == -1) {break;}if (i < stack[top]) {if (chars[i] < chars[stack[top]]) {char temp = chars[i];chars[i] = chars[stack[top]];chars[stack[top]] = temp;return Integer.parseInt(new String(chars));}} else {top--;}}return num;}
}
cpp
class Solution {
public:int maximumSwap(int num) {std::string numStr = std::to_string(num);int n = numStr.length();std::vector<int> stack;for (int i = n - 1; i >= 0; i--) {if (stack.empty() || numStr[i] > numStr[stack.back()]) {stack.push_back(i);}}for (int i = 0; i < n; i++) {if (stack.empty()) {break;}if (i < stack.back()) {if (numStr[i] < numStr[stack.back()]) {std::swap(numStr[i], numStr[stack.back()]);return std::stoi(numStr);}} else {stack.pop_back();}}return num;}
};
时空复杂度
时间复杂度:O(N)
。仅需一次遍历所有数字。
空间复杂度:O(1)
。栈所占空间,最大为9
,可视为常数级别空间。
华为OD算法/大厂面试高频题算法练习冲刺训练
-
华为OD算法/大厂面试高频题算法冲刺训练目前开始常态化报名!目前已服务100+同学成功上岸!
-
课程讲师为全网50w+粉丝编程博主@吴师兄学算法 以及小红书头部编程博主@闭着眼睛学数理化
-
每期人数维持在20人内,保证能够最大限度地满足到每一个同学的需求,达到和1v1同样的学习效果!
-
60+天陪伴式学习,40+直播课时,300+动画图解视频,300+LeetCode经典题,200+华为OD真题/大厂真题,还有简历修改、模拟面试、专属HR对接将为你解锁
-
可上全网独家的欧弟OJ系统练习华子OD、大厂真题
-
可查看链接 大厂真题汇总 & OD真题汇总(持续更新)
-
绿色聊天软件戳
od1336
了解更多
相关文章:

【Py/Java/C++三种语言详解】LeetCode每日一题240122【贪心】LeetCode670、最大交换
文章目录 题目链接题目描述解题思路为什么是贪心一个带图的例子 代码pythonjavacpp时空复杂度 华为OD算法/大厂面试高频题算法练习冲刺训练 题目链接 LeetCode670、最大交换 题目描述 给定一个非负整数数组 nums 和一个整数 k ,你需要将这个数组分成 k 个非空的连…...

Linux/Doctor
Enumeration nmap 已知目标开放了22,80,8089端口,扫描详细情况如下 可以看到对外开放了22,80,8089三个端口 TCP/80 SSTI 访问80端口,有一个infodoctors.htb的电子邮件,点击其他的也没有什么反应,猜测有可能需要域名访问 在/et…...

嵌入式linux学习之系统烧录
1.所需文件 1. 开发板为正点原子stm32mp157,文件可按照linux驱动教程编译,也可在正点原子文档->08、系统镜像\02、出厂系统镜像中找到: 2.烧录 1.拨码开关为000(usb启动),otg接口接入虚拟机,打开stm32cubeProgrammer: 2.页面…...

JVM-初始JVM
什么是JVM JVM 全称是 Java Virtual Machine,中文译名 Java虚拟机。JVM 本质上是一个运行在计算机上的程序,他的职责是运行Java字节码文件。 Java源代码执行流程如下: JVM的功能 1 - 解释和运行 2 - 内存管理 3 - 即时编译 解释和运行 解释…...
EXCEL VBA网抓技巧-复制网页表格,不用遍历单元格
EXCEL VBA网抓技巧-复制网页表格,不用遍历单元格 对应表格复制 Sub tableTest()Set winhttp CreateObject("winhttp.WinHttpRequest.5.1")Set HTML CreateObject("htmlfile")Set oWindow HTML.ParentWindowUrl "https://www.taiwanlo…...

动态规划——炮兵回城【集训笔记】
题目描述 游戏盘面是一个m行n列的方格矩阵,将每个方格用坐标表示,行坐标从下到上依次递增,列坐标从左至右依次递增,左下角方格的坐标为(1,1),则右上角方格的坐标为(m,n)。 游戏结束盘上只剩下一枚炮兵没有回到城池中&a…...

低成本扫码点餐:1000元全包
在数字化时代,扫码点餐已经成为餐饮行业的标配。然而,对于许多小规模或初创的餐饮企业来说,开发一套完整的扫码点餐系统是一项成本高昂的任务。今天,我们将向您介绍一个低成本、高效的方法,让您用1000块钱轻松搞定一套…...

五款焊在电脑上的效率软件
在当今快节奏的商业环境中,提高工作效率成为了每个人都渴望实现的目标。尤其是在面对繁忙的工作日程、庞杂的任务清单和团队合作的压力时,我们需要一些可靠的工具来帮助我们更好地管理时间、组织工作和提高生产力。幸运的是,现在有许多高效的…...

小程序样例3:根据日历创建待办事项
基本功能 1、待办事项查看 选择不同的日期显示不同的待办: 2、选择日期后 新增事项: 3. 点击事项,查看详情 4、删除事项:删除事项3之后,剩余事项2 5、点击日期可以选择更多的月: 实现思路: 1、数据结构&a…...

计算机设计大赛 协同过滤电影推荐系统
文章目录 1 简介1 设计概要2 课题背景和目的3 协同过滤算法原理3.1 基于用户的协同过滤推荐算法实现原理3.1.1 步骤13.1.2 步骤23.1.3 步骤33.1.4 步骤4 4 系统实现4.1 开发环境4.2 系统功能描述4.3 系统数据流程4.3.1 用户端数据流程4.3.2 管理员端数据流程 4.4 系统功能设计 …...
docker下安装rabbitmq
1.查询rabbitmq的镜像 docker search rabbitmq 2.安装镜像 如果需要安装其他版本在rabbitmq后面跟上版本号即可 docker pull rabbitmq:3.7.7-management docker pull rabbitmq:版本号 -management 直接安装最新的 docker pull rabbitmq 3.启动容器 docker run -dit --name rab…...

量子网络是什么
量子网络是基于量子力学规律对量子信息进行存储、处理和传输的物理装置,是实现量子通讯和大规模量子计算的基础。清华大学研究团队利用同种离子的双类型量子比特编码,在国际上首次实现无串扰的量子网络节点,对未来实现量子通讯和大规模量子计…...

使用javadoc生成maven项目的文档
概述:Maven 提供了 javadoc 插件来执行这个任务。 废话不多说,让我们开始操作吧!!! 第一步:引入插件 在 pom.xml 中配置 javadoc 插件: 在 Maven 项目的 pom.xml 文件中,你需要添加…...

移动端 h5-table react版本支持虚拟列表
介绍 适用于 react ts 的 h5 移动端项目 table 组件 github 链接 :https://github.com/duKD/react-h5-table 有帮助的话 给个小星星 有两种表格组件 常规的: 支持 左侧固定 滑动 每行点击回调 支持 指定列排序 支持滚动加载更多 效果和之前写的vue…...

解决Windows系统本地端口被占用的问题
一、解决Windows系统本地端口被占用的问题,首先我们要在虚拟机上人为的占用本地端口 二、占用端口方法:以管理员身份运行cmd;输入net stop http;如果提示是否真的需要停止这些服务,则选择“Y”;完成后输入:sc config http startdisabled 弹出上图内容则成…...

(超全七大错误)Invalid bound statement (not found): com.xxx.dao.xxxDao.add
1.确保你把dao和mapper都在applicationContext.xml中都扫描了 xml文件 <bean id"sqlSessionFactory" class"org.mybatis.spring.SqlSessionFactoryBean"><property name"dataSource" ref"dataSource"/><property nam…...

【操作系统】实验八 proc文件系统
🕺作者: 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux 😘欢迎关注:👍点赞🙌收藏✍️留言 🏇码字不易,你的👍点赞🙌收藏❤️关注对我真的很重要&…...

基于RMF的信贷风控标签客户分层管理
根据美国 数据库营销研究所Arthur Hughes的研究,客户数据库中有3个神奇的要素,这3个要素构成了数据分析最好的指标:最近一次消费 (Recency)、消费频率(Frequency)、消费金额 (Monetary)。这就是RMF模型,RMF模型是用户分层的重要手…...

【MySQL】如何通过DDL去创建和修改员工信息表
🌈个人主页: Aileen_0v0 🔥热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 💫个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-fmKISDBsFq74ab2Z {font-family:"trebuchet ms",verdana,arial,sans-serif;font-siz…...

Spring 事务原理一
从本篇博客开始,我们将梳理Spring事务相关的知识点。在开始前,想先给自己定一个目标:通过此次梳理要完全理解事务的基本概念及Spring实现事务的基本原理。为实现这个目标我想按以下几个步骤进行: 讲解事务中的一些基本概念使用Sp…...

深入剖析AI大模型:大模型时代的 Prompt 工程全解析
今天聊的内容,我认为是AI开发里面非常重要的内容。它在AI开发里无处不在,当你对 AI 助手说 "用李白的风格写一首关于人工智能的诗",或者让翻译模型 "将这段合同翻译成商务日语" 时,输入的这句话就是 Prompt。…...

【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器
一.自适应梯度算法Adagrad概述 Adagrad(Adaptive Gradient Algorithm)是一种自适应学习率的优化算法,由Duchi等人在2011年提出。其核心思想是针对不同参数自动调整学习率,适合处理稀疏数据和不同参数梯度差异较大的场景。Adagrad通…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
VTK如何让部分单位不可见
最近遇到一个需求,需要让一个vtkDataSet中的部分单元不可见,查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行,是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示,主要是最后一个参数,透明度…...
工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配
AI3D视觉的工业赋能者 迁移科技成立于2017年,作为行业领先的3D工业相机及视觉系统供应商,累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成,通过稳定、易用、高回报的AI3D视觉系统,为汽车、新能源、金属制造等行…...

智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...

深入浅出深度学习基础:从感知机到全连接神经网络的核心原理与应用
文章目录 前言一、感知机 (Perceptron)1.1 基础介绍1.1.1 感知机是什么?1.1.2 感知机的工作原理 1.2 感知机的简单应用:基本逻辑门1.2.1 逻辑与 (Logic AND)1.2.2 逻辑或 (Logic OR)1.2.3 逻辑与非 (Logic NAND) 1.3 感知机的实现1.3.1 简单实现 (基于阈…...
比较数据迁移后MySQL数据库和OceanBase数据仓库中的表
设计一个MySQL数据库和OceanBase数据仓库的表数据比较的详细程序流程,两张表是相同的结构,都有整型主键id字段,需要每次从数据库分批取得2000条数据,用于比较,比较操作的同时可以再取2000条数据,等上一次比较完成之后,开始比较,直到比较完所有的数据。比较操作需要比较…...
MinIO Docker 部署:仅开放一个端口
MinIO Docker 部署:仅开放一个端口 在实际的服务器部署中,出于安全和管理的考虑,我们可能只能开放一个端口。MinIO 是一个高性能的对象存储服务,支持 Docker 部署,但默认情况下它需要两个端口:一个是 API 端口(用于存储和访问数据),另一个是控制台端口(用于管理界面…...