Python算法例19 创建最大数

1. 问题描述

给定两个长度分别是m和n的数组,数组的每个元素都是数字0~9,从这两个数组当中选出k个数字来创建一个最大数,其中k满足k<=m+n,选出来的数字在创建最大数里的位置必须与在原数组内的相对位置一致。返回k个元素的整数数组,尽可能优化算法的时间复杂度和空间复杂度。

2. 问题示例

给出nums1=[3,4,6,5],nums2=[9,1,2,5,8,3],k=5,返回[9,8,6,5,3];给出nums1=[6,7],nums2=[6,0,4],k=5,返回[6,7,6,0,4];给出nums1=[3,9],nums2=[8,9],k=3,返回[9,8,9]。

3. 代码实现

使用贪心算法来解决。具体来说,我们可以固定从nums1中选取多少个数字,然后从nums2中选取剩余的数字,使得它们合起来构成一个长度为k的最大数。

为了实现这个算法,我们需要确定以下步骤:

  1. 实现一个函数,将一个数组中的元素重排,使得它们构成的数字最大;
  2. 对于一个给定的数组nums和整数k,我们需要找到由nums中的数字组成的长度为k的最大数;
  3. 给定两个数组nums1和nums2以及整数k,我们需要找到由nums1和nums2中的数字组成的长度为k的最大数。
def maxNumber(nums1, nums2, k):# 重排数组中的元素,使得它们构成的数字最大def merge(A, B):return [max(A, B).pop(0) for _ in A+B]# 找到由nums中的数字组成的长度为k的最大数def prepare(nums, k):drop = len(nums) - kout = []for num in nums:while drop and out and out[-1] < num:out.pop()drop -= 1out.append(num)return out[:k]# 找到由nums1和nums2中的数字组成的长度为k的最大数def lexicographically_greater(a, i, b, j):while i < len(a) and j < len(b) and a[i] == b[j]:i += 1j += 1return j == len(b) or (i < len(a) and a[i] > b[j])# 尝试从nums1中选取i个数字,从nums2中选取k-i个数字res = []for i in range(max(0, k-len(nums2)), min(k, len(nums1))+1):candidate = merge(prepare(nums1, i), prepare(nums2, k-i))res = max(candidate, res) if res != [] else candidatereturn res
nums1 = [3, 4, 6, 5]
nums2 = [9, 1, 2, 5, 8, 3]
k = 5
result = maxNumber(nums1, nums2, k)
print(result)  # 输出 [9, 8, 6, 5, 3]

其中,函数merge()用于将两个数组合并成一个最大数,函数prepare()用于找到由一个数组中的数字组成的长度为k的最大数,函数lexicographically_greater()用于比较两个数组的字典序大小。具体来说,这个函数的作用是,给定两个数组a和b以及它们的起始下标i和j,如果在i和j之后,a比b字典序更大,则返回True,否则返回False。

在主函数maxNumber()中,我们枚举从nums1中选取的数字的个数i,然后从nums2中选取剩余的数字k-i,将它们合并成一个最大数candidate。最后,我们将所有生成的最大数中的最大者返回即可。这个算法的时间复杂度为O((m+n)^3),空间复杂度为O(m+n)。

采用动态规划的方法。具体来说,我们可以使用两个动态规划数组来保存最大数的信息,以减少重复计算。

def maxNumber(nums1, nums2, k):def get_max_subsequence(nums, k):stack = []pop_count = len(nums) - kfor num in nums:while pop_count and stack and stack[-1] < num:stack.pop()pop_count -= 1stack.append(num)return stack[:k]def merge(nums1, nums2):merged = []i, j = 0, 0while i < len(nums1) and j < len(nums2):if nums1[i:] > nums2[j:]:merged.append(nums1[i])i += 1else:merged.append(nums2[j])j += 1merged.extend(nums1[i:])merged.extend(nums2[j:])return mergedresult = []for i in range(max(0, k-len(nums2)), min(k, len(nums1))+1):subsequence1 = get_max_subsequence(nums1, i)subsequence2 = get_max_subsequence(nums2, k-i)merged = merge(subsequence1, subsequence2)result = max(result, merged)return result
nums1 = [6,7]
nums2 = [6,0,4]
k = 5
result = maxNumber(nums1, nums2, k)
print(result)

在这个实现中,我们定义了两个辅助函数:get_max_subsequence()和merge()。函数get_max_subsequence()用于从一个数组中获取长度为k的最大子序列,这个函数的实现与之前贪心算法中的prepare()函数相同。函数merge()用于合并两个子序列,这个函数的实现与之前贪心算法中的merge()函数相同。

在主函数maxNumber()中,我们使用动态规划来生成最大数。我们首先枚举从nums1中选取的数字的个数i,然后使用get_max_subsequence()函数分别从nums1和nums2中获取长度为i和k-i的最大子序列。接下来,我们使用merge()函数将这两个子序列合并成一个最大数,并将其与之前的结果比较,保留更大的那个。最后,返回最终的最大数。这个算法的时间复杂度为O((m+n)^2),空间复杂度为O(m+n)。

使用单调栈的思想来减少重复计算。具体来说,我们可以使用两个单调递减栈,分别从nums1和nums2中构建。栈的长度为k,每次入栈时,判断当前元素与栈顶元素的大小关系,如果当前元素更大,则将栈顶元素出栈,直到满足栈的长度要求或者栈为空。然后将当前元素入栈。

def maxNumber(nums1, nums2, k):def build_max_stack(nums, k):stack = []drop_count = len(nums) - kfor num in nums:while drop_count > 0 and stack and stack[-1] < num:stack.pop()drop_count -= 1stack.append(num)return stack[:k]def merge(nums1, nums2):merged = []i, j = 0, 0while i < len(nums1) and j < len(nums2):if nums1[i:] > nums2[j:]:merged.append(nums1[i])i += 1else:merged.append(nums2[j])j += 1merged.extend(nums1[i:])merged.extend(nums2[j:])return mergedresult = []for i in range(max(0, k - len(nums2)), min(k, len(nums1)) + 1):subsequence1 = build_max_stack(nums1, i)subsequence2 = build_max_stack(nums2, k - i)merged = merge(subsequence1, subsequence2)result = max(result, merged)return result
nums1 = [3,9]
nums2 = [8,9]
k = 3
result = maxNumber(nums1, nums2, k)
print(result)

在这个实现中,我们使用了两个辅助函数:build_max_stack()和merge()。函数build_max_stack()用于构建单调递减栈,获取长度为k的最大子序列。函数merge()用于合并两个子序列。

在主函数maxNumber()中,我们通过枚举从nums1中选取的数字的个数i,分别构建nums1和nums2的单调递减栈,并通过merge()函数将这两个子序列合并成一个最大数。最后,返回最终的最大数。这个算法的时间复杂度为O(m+n+k(m+n)),其中m和n分别是nums1和nums2的长度。

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

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

相关文章

Git报错x509: certificate signed by unknown authority

下载报错&#xff1a; Error downloading object: model-00001-of-00008.safetensors (ed3ac49): Smudge error: Error downloading model-00001-of-00008.safetensors (ed3ac4983f682a999b0e4b6f072aad294c4fd9a7e968e90835ba5c4b466d3c7c): LFS: Get https://cdn-lfs.huggin…

TDengine 创始人陶建辉受邀参与 TOP100Summit,发表工程师文化主题演讲

在 AGI 时代&#xff0c;数字化成为组织形态的重要特征&#xff0c;它可以帮助组织实现上下一致的目标和信息的高频传递&#xff0c;从而实现战略目标的协同和敏捷进化。在这样的大背景下&#xff0c;开发者们面临的实际挑战是如何避免技术和业务之间的割裂。 12 月 14-17 日&…

Google Play不会凭空消失,这篇文章带你重新找回丢失的它

你是不是因为不小心从手机上删除了Google Play而难过?或者你是否注意到你的Android设备上缺少Google Play图标?你一定很担心你现在会如何下载应用程序。别担心。在这篇文章中,我们将告诉你如何恢复已删除的谷歌商店。 Google Play可以卸载吗 让我们明确一点:除了一些特殊…

若依(ruoyi)管理系统标题和logo修改

1、网页上的logo 2、页面中的logo 进入ruoyi-ui --> src --> assets --> logo --> logo.png&#xff0c;把这个图片换成你自己的logo 3、网页标题 进入ruoyi-ui --> src --> layout --> components --> Sidebar --> Logo.vue&#xff0c;将里面的…

如何在centos装maven

1&#xff1a;进入maven官方网址&#xff1a; Maven – Welcome to Apache Mavenhttps://maven.apache.org/index.html点击 然后找一个版本&#xff0c;比如3.8.2 centos点击这个&#xff1a; 将这个下下来后到虚拟机里解压&#xff0c;tar -zxvf 这个gz文件 然后配置环境变…

Java多线程,一文掌握Java多线程知识文集。

&#x1f3c6;作者简介&#xff0c;普修罗双战士&#xff0c;一直追求不断学习和成长&#xff0c;在技术的道路上持续探索和实践。 &#x1f3c6;多年互联网行业从业经验&#xff0c;历任核心研发工程师&#xff0c;项目技术负责人。 &#x1f389;欢迎 &#x1f44d;点赞✍评论…

Redis底层数据结构原理

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码、Kafka原理、分布式技术原理、数据库技术&#x1f525;如果感觉博主的文章还不错的…

el-table设置默认选中报错_this.$refs.singleTable.toggleAllSelection is not a function

直接使用以下的方法&#xff0c;报错信息是_this.$refs.singleTable.toggleAllSelection is not a function this.$refs.singleTable.toggleAllSelection()看了网上的解决方法&#xff0c;加了this.$nextTick,代码如下&#xff0c;但还是报错Error in nextTick: "TypeErr…

工业互联网平台存在意义是什么?国内有哪些工业互联网平台?

一、工业互联网平台存在意义是什么&#xff1f; 工业互联网平台是一个连接设备与服务、数据与人的跨行业、跨领域的全新工业平台。工业互联网平台利用了互联网、物联网、大数据、AI等技术&#xff0c;集成各类工业设备&#xff0c;不断采集和分析数据&#xff0c;以实现设备状…

Python算法例20 最接近的k个数

1. 问题描述 给定一个目标数target&#xff0c;一个非负整数k&#xff0c;一个按照升序排列的数组A。在A中找出与target最接近的k个整数&#xff0c;返回这k个数并按照与target的接近程度从小到大排序&#xff0c;如果接近程度相当&#xff0c;那么值小的排在前面。 2. 问题示…

测试用例的修改更新

测试用例的修改更新是指测试过程中由于用户需求的改变&#xff0c;或者测试过程中发现有新的需求产生&#xff0c;使得测试用例需要进行修改。修改更新测试用例不仅是一种测试技术&#xff0c;更是一种质量保证的方法。但修改和更新测试用例的技术要点在于&#xff1a; 1、执行…

java实现(燃油车车牌生成)

目录 暗箱 常量车牌特殊车牌 车牌规则 1、随机生成五位数字车牌号 2、随机生成车牌数字字母或者全数字 3、随机生成车牌数字字母(两位字母紧靠&#xff0c;三位数字紧靠) 4、指定车牌号码(数字加字母) 5、购买VIP号码 炸弹号&#xff0c;顺子号 6、车牌规则(是否是AABA…