【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

从这个例子可以看出贪心的策略是:

  1. 首选一个尽可能大的数字(比如示例中选择字符"9"
  2. 如果有多个最大的数字,则优先选择位置尽可能靠后的那个(比如示例中选择第三个"9"
  3. 将该数字交换到尽可能靠前的位置,即交换到第一个小于该数字的位置(比如示例中索引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]],说明此时可以交换位置istack[-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了解更多

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

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

相关文章

电信联通5G共建共享方案实施及验证

一、情况概述 随着2019年9月9日中国电信集团与联通签署《5G网络共建共享框架合作协议书》&#xff0c;电信与联通在全国范围内合作共建5G接入网络。根据合作协议&#xff0c;联通运营公司将与中国电信在全国范围内合作共建一张5G接入网络, 双方划定区域&#xff0c;分区建设&a…

C++-QT-QString -CString -string 互转

网上常用的函数在环境&#xff08;VS2022 ATL包含QT库的项目&#xff09;中转换不了。 1.QString 转String std::string str qstr.toStdString(); //不行 QString qstr "Hello, world!";//1. 将QString转换为std::string 不行 //std::string str qstr.toSt…

分享多种vcruntime140_1.dll丢失修复办法,vcruntime140_1.dll文件下载

vcruntime140_1.dll是Windows操作系统中的一个重要系统文件&#xff0c;它与C运行库相关。当计算机上缺少或损坏了vcruntime140_1.dll文件时&#xff0c;可能会导致一系列问题和错误。出现这文件错误&#xff0c;应该很多小伙伴都会想到重新下载vcruntime140_1.dll&#xff0c;…

GitHub 打不开问题解决

1、找到记事本&#xff0c;以管理员身份运行 2、打开记事本后点击左上角文件下的打开&#xff0c;然后找到 C:\Windows\System32\drivers\etc 文件夹下的 hosts 文件 这里需要注意的是右下角文件类型要选择所有文件类型才能看到 hosts 文件。 3、打开文件后&#xff0c;在最下…

AI智能绘图

AI智能绘图是一种创新的图像生成技术&#xff0c;它使用人工智能算法来根据用户输入的文本描述或参考图片自动生成艺术作品。这种技术分为两个方面&#xff1a;文生图和图生图。 首先我们需要浏览器搜索“固乔科技”官网&#xff0c;先下载并安装固乔智创助手软件。完成后&…

【vueCms】vuecms_xg安装与使用视频讲解【视频讲解】

克隆项目 gitee项目地址:https://gitee.com/derekgo/vue-cms_xg git clone gitgitee.com:derekgo/vue-cms_xg.git开启自动化脚本 点击后&#xff0c;运行http://localhost:3002 本地配置 开启本地redis 开启redis redis-server.exe redis.windows.conf重启redis 注意:…

matlab模型变量一般说明,标定和显示量,以及产生a2l文件,自动填充a2l地址,并使用标定工具ati进行标定(推荐重要)

注意我是用的是matlab2019b 1&#xff0c;输入标定量&#xff0c;使用constant&#xff0c;用cal函数包裹 2&#xff0c;输出显示量&#xff0c;在划线上标注&#xff0c;然后用display函数包裹&#xff0c; 第一步和第二步完成以后&#xff0c;生产标定量a2l 3&#xff0c;输入…

什么是护网行动?

护网&#xff0c;也称“网络保护”&#xff0c;是指网络安全人员对企业或组织的网络进行检查、维护和保护&#xff0c;以防止网络受到黑客攻击、病毒、木马或其它恶意程序的侵入和损害。护网工作包括&#xff1a;网络安全规划、网络配置和控制、漏洞发现和修复、入侵检测和防范…

Qt 多次绘图

使用Qt 的时候发现&#xff1a; 背景&#xff1a;自己定义一个类&#xff0c;把它和某个ui文件绑定。(类似 Qt creator 默认创建的工程&#xff09;问题&#xff1a;当鼠标在窗口内单击的时候会触发2次绘图。&#xff1f;难道不应该是一次吗&#xff1f; 于是开始了如下的测试…

电源设计如果只看电压跌落,不看电流密度会怎么样?

高速先生成员--黄刚 大家知道&#xff0c;电源直流设计的理论其实非常的简单&#xff0c;归根到底就是欧姆定律&#xff0c;电源芯片给负载供给电流&#xff0c;电流经过传输路径有一定的压降&#xff0c;最终到达负载端的电压值就是我们接收芯片关注的结果。 在PCB设计中&…

新网站收录需要多长时间完成审核

新网站的收录时间因多种因素而异。 一般来说&#xff0c;新上线的网站可能在最快3-7天内被百度收录&#xff0c;尤其是那些有高质量内容的网 然而&#xff0c;通常情况下&#xff0c;新网站的收录可能会在7-15天左右发生。 有些情况下&#xff0c;如果网站的内容足够丰富和有价…

nvm 配置淘宝镜像失效,以及安装node后 npm-v 无效

win11 nvm版本 1.1.4 和1.1.7和1.1.12&#xff08;目前最新版本24年 一月二十三日&#xff09; 以上nvm版本都会出现一下问题&#xff0c; 从https://github.com/coreybutler/nvm-windows/releases 下载nvm安装包如下图 傻瓜式安装后&#xff0c;不用去配置环境变量&#…

【2023】无痕制作钓鱼邮件

当涉及到网络安全和信息保护时&#xff0c;钓鱼攻击一直是备受关注的话题。钓鱼攻击是一种利用社交工程手段欺骗用户&#xff0c;诱使其提供敏感信息或执行恶意操作的攻击方式。它不仅威胁个人用户的隐私和财产安全&#xff0c;也对组织和企业的机密信息构成严重威胁。为了增强…

Linux第35步_在“移植uboot”前安装libncurses5-dev

在“移植uboot”前&#xff0c;需要在Ubuntu中安装“libncurses5-dev”&#xff0c;否则在“编译uboot”时&#xff0c;会报错。目的是保证顺利移植“uboot”。 1、打开终端 2、输入“sudo apt-get install libncurses5-dev bison flex回车”&#xff1b; 3、输入密码“1234…

3W Star 网易云音乐第三方开源 API 仓库因侵权被要求删除

NeteaseCloudMusicApi是一个使用Node.js编写的非官方网易云音乐API&#xff0c;用于获取网易云音乐平台的歌曲信息。该项目是完全开源的&#xff0c;在GitHub上获得了超过3万的star。 根据公开信息&#xff0c;NeteaseCloudMusicApi的主要目的是整理网易云音乐公开的网页接口&a…

Linux逻辑卷创建与扩容

​ 钢铁知识库&#xff0c;一个学习python爬虫、数据分析的知识库。在职运维&#xff0c;又喜欢开发。 什么是lvm逻辑卷 ​ LVM&#xff08;逻辑卷管理&#xff09;是一种在Linux环境中广泛使用的磁盘管理技术。它允许管理员创建逻辑上的磁盘分区&#xff0c;这些分区可以跨越…

node 第二十三天 mongoDB shell 命令 CRUD 增删改查 基础

什么是 mongoDB shell 命令 mongoDB shell 命令就是在cmd窗口或者powershell窗口与mongoDB交互的命令, 以下简称mongosh 对应我们上一天安装的 mongosh 工具 有什么用 mongosh 对一般的开发者可能意义不大, 因为在开发过程中我们会基于某一款语言来使用mongoDB, 比如在node端我…

需求分析的结构化和面向对象方法

需求分析的方法&#xff0c;主要有结构化&#xff08;SA&#xff09;方法、面向对象分析&#xff08;OOA&#xff09;方法。另外有面向问题域的分析(Problem Domain Oriented Analysis&#xff0c;PDOA)方法&#xff0c;还有一些形式化方法例如VDM ( Vienna Design Method)和Z等…

基于cubeMX的正点原子miniSTM32对W25Q64的存储使用

一、实现目标 使用cubeMX建立项目工程&#xff0c;结合正点原子提供的hal库对W25Q64闪存调用的例程&#xff0c;实现W25Q64的读写。 二、实现过程 1、首先建立cubeMX工程&#xff0c;其他项设置不再叙述&#xff0c;只看连接W25Q64的SPI设置&#xff0c;这里使用SPI1&#xf…

2.1 关系模型的数据结构及形式化定义

2.1.1 关系 单一的数据结构----关系 现实世界的实体以及实体间的各种联系均用关系来表示 。 逻辑结构----二维表 从用户角度&#xff0c;关系模型中数据的逻辑结构是一张二维表 。 建立在集合代数的基础上。 1. 域&#xff08;domain&#xff09; 2. 笛卡儿积&#xff08;C…