拼接最大数
程序员文章站
2022-03-11 09:22:29
给定长度分别为 m 和 n 的两个数组,其元素由 0-9 构成,表示两个自然数各位上的数字。现在从这两个数组中选出 k (k <= m + n) 个数字拼接成一个新的数,要求从同一个数组中取出的数字保持其在原数组中的相对顺序。求满足该条件的最大数。结果返回一个表示该最大数的长度为 k 的数组。说明: 请尽可能地优化你算法的时间和空间复杂度。示例 1:输入:nums1 = [3, 4, 6, 5]nums2 = [9, 1, 2, 5, 8, 3]k = 5输出:[9, 8, 6, 5...
给定长度分别为 m
和 n
的两个数组,其元素由 0-9
构成,表示两个自然数各位上的数字。现在从这两个数组中选出 k (k <= m + n)
个数字拼接成一个新的数,要求从同一个数组中取出的数字保持其在原数组中的相对顺序。
求满足该条件的最大数。结果返回一个表示该最大数的长度为 k
的数组。
说明: 请尽可能地优化你算法的时间和空间复杂度。
示例 1:
输入:
nums1 = [3, 4, 6, 5]
nums2 = [9, 1, 2, 5, 8, 3]
k = 5
输出:
[9, 8, 6, 5, 3]
示例 2:
输入:
nums1 = [6, 7]
nums2 = [6, 0, 4]
k = 5
输出:
[6, 7, 6, 0, 4]
示例 3:
输入:
nums1 = [3, 9]
nums2 = [8, 9]
k = 3
输出:
[9, 8, 9]
解答
这题不会,搬运下官方解答吧。思路还是比较清晰的,使用单调栈维护从两个数组中抽取的相应长度的最大子序列,再进行拼接,针对两个数组中抽取子序列的不同长度进行遍历找到最大解:
class Solution {
public:
vector<int> maxNumber(vector<int>& nums1, vector<int>& nums2, int k) {
int m = nums1.size(), n = nums2.size();
vector<int> maxSubsequence(k, 0);
// 从nums1中抽取子序列的最小和最大长度
int start = max(0, k - n), end = min(k, m);
for (int i = start; i <= end; i++) {
vector<int> subsequence1(MaxSubsequence(nums1, i));
vector<int> subsequence2(MaxSubsequence(nums2, k - i));
vector<int> curMaxSubsequence(merge(subsequence1, subsequence2));
if (compare(curMaxSubsequence, 0, maxSubsequence, 0) > 0) {
maxSubsequence.swap(curMaxSubsequence);
}
}
return maxSubsequence;
}
// 使用数组实现单调栈
// 也不一定完全是递减序列,例如[1, 2, 3, 4]中取两个数,应该是[3, 4]
// remain的作用:以[3, 2, 1, 4, 5]为例,取3个数应该返回[3, 4, 5]
// remain初始值为2,循环到元素4时,栈内维持单调递减,即栈内元素为[3, 2, 1]
// 此时remain值为2,如果是完全的单调栈的话,遍历到4以后就会舍弃掉此前的3个元素
// 这样会使最后返回的栈元素总数小于3,remain表示最多可以从当前栈中剔除几个元素
// 因为要保证最后返回元素总数恒定
vector<int> MaxSubsequence(vector<int>& nums, int k) {
int length = nums.size();
vector<int> stack(k, 0);
int top = -1;
int remain = length - k;
for (int i = 0; i < length; i++) {
int num = nums[i];
while (top >= 0 && stack[top] < num && remain > 0) {
top--;
remain--;
}
if (top < k - 1) {
stack[++top] = num;
} else {
remain--;
}
}
return stack;
}
// 按照字典序进行比较
vector<int> merge(vector<int>& subsequence1, vector<int>& subsequence2) {
int x = subsequence1.size(), y = subsequence2.size();
if (x == 0) {
return subsequence2;
}
if (y == 0) {
return subsequence1;
}
int mergeLength = x + y;
vector<int> merged(mergeLength);
int index1 = 0, index2 = 0;
for (int i = 0; i < mergeLength; i++) {
if (compare(subsequence1, index1, subsequence2, index2) > 0) {
merged[i] = subsequence1[index1++];
} else {
merged[i] = subsequence2[index2++];
}
}
return merged;
}
int compare(vector<int>& subsequence1, int index1, vector<int>& subsequence2, int index2) {
int x = subsequence1.size(), y = subsequence2.size();
while (index1 < x && index2 < y) {
int difference = subsequence1[index1] - subsequence2[index2];
if (difference != 0) {
return difference;
}
index1++;
index2++;
}
return (x - index1) - (y - index2);
}
};
// 作者:LeetCode-Solution
// 链接:https://leetcode-cn.com/problems/create-maximum-number/solution/pin-jie-zui-da-shu-by-leetcode-solution/
// 来源:力扣(LeetCode)
// 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
本文地址:https://blog.csdn.net/WhiteGray/article/details/110470918
推荐阅读