2021-07-30:两个有序数组间相加和的Topk问题。给定两个有序数组arr1和arr2,再给定一个整数k,返回来自arr1和arr2的两个数相加和最大的前k个,两个数必须分别来自两个数组。按照降
2021-07-30:两个有序数组间相加和的Topk问题。给定两个有序数组arr1和arr2,再给定一个整数k,返回来自arr1和arr2的两个数相加和最大的前k个,两个数必须分别来自两个数组。按照降序输出。[要求]时间复杂度为O(klogk)。
福大大 答案2021-07-30:
1.左神方法。大根堆。
时间复杂度:O(klogk)。
空间复杂度:O(k)。
2.我的方法。小根堆。两个有序数组构成一个二维数组。然后从右下往左上遍历,当遍历数量大于等于k时,停止遍历。见图。
时间复杂度:略大于O(k)。
空间复杂度:O(k)。

代码用golang编写。代码如下:
package main
import (
"fmt"
"sort"
)
func main() {
arr1 := []int{1, 2, 3, 4, 5}
arr2 := []int{3, 5, 7, 9, 11}
topK := 4
if true {
ret := topKSum1(arr1, arr2, topK)
fmt.Println("左神的方法:", ret)
}
if true {
ret := topKSum2(arr1, arr2, topK)
fmt.Println("我的方法:", ret)
}
}
type Node struct {
index1 int // arr1中的位置
index2 int // arr2中的位置
sum int // arr1[index1] + arr2[index2]的值
}
func NewNode(i1 int, i2 int, s int) *Node {
ret := &Node{}
ret.index1 = i1
ret.index2 = i2
ret.sum = s
return ret
}
func topKSum1(arr1 []int, arr2 []int, topK int) []int {
if len(arr1) == 0 || len(arr2) == 0 || topK < 1 {
return nil
}
N := len(arr1)
M := len(arr2)
topK = getMin(topK, N*M)
res := make([]int, topK)
resIndex := 0
maxHeap := make([]*Node, 0)
set := make(map[int]struct{})
i1 := N - 1
i2 := M - 1
Push(&maxHeap, NewNode(i1, i2, arr1[i1]+arr2[i2]))
set[x(i1, i2, M)] = struct{}{}
for resIndex != topK {
curNode := Pop(&maxHeap)
res[resIndex] = curNode.sum
resIndex++
i1 = curNode.index1
i2 = curNode.index2
delete(set, x(i1, i2, M))
_, ok := set[x(i1-1, i2, M)]
if i1-1 >= 0 && !ok {
set[x(i1-1, i2, M)] = struct{}{}
Push(&maxHeap, NewNode(i1-1, i2, arr1[i1-1]+arr2[i2]))
}
_, ok = set[x(i1, i2-1, M)]
if i2-1 >= 0 && !ok {
set[x(i1, i2-1, M)] = struct{}{}
Push(&maxHeap, NewNode(i1, i2-1, arr1[i1]+arr2[i2-1]))
}
}
return res
}
func getMin(a int, b int) int {
if a < b {
return a
} else {
return b
}
}
func x(i1 int, i2 int, M int) int {
return i1*M + i2
}
func Push(maxHeap *[]*Node, node *Node) {
*maxHeap = append(*maxHeap, node)
}
func Pop(maxHeap *[]*Node) *Node {
sort.Slice(*maxHeap, func(i, j int) bool {
return (*maxHeap)[i].sum < (*maxHeap)[j].sum
})
ans := (*maxHeap)[len(*maxHeap)-1]
*maxHeap = (*maxHeap)[0 : len(*maxHeap)-1]
return ans
}
func topKSum2(arr1 []int, arr2 []int, topK int) []int {
if len(arr1) == 0 || len(arr2) == 0 || topK < 1 {
return nil
}
N := len(arr1)
M := len(arr2)
topK = getMin(topK, N*M)
maxHeap := make([]int, 0)
i1Start, i2Start := N-1, M-1
count := 0
for {
for i1, i2 := i1Start, i2Start; i1 >= 0 && i2 < M; i1, i2 = i1-1, i2+1 {
PushInt(&maxHeap, arr1[i1]+arr2[i2])
if len(maxHeap) > topK {
PopInt(&maxHeap)
}
count++
}
if count >= topK {
break
}
if i2Start > 0 { //左移
i2Start--
} else { //上移
i1Start--
}
}
return maxHeap
}
func PushInt(maxHeap *[]int, node int) {
*maxHeap = append(*maxHeap, node)
}
func PopInt(maxHeap *[]int) int {
sort.Slice(*maxHeap, func(i, j int) bool {
return (*maxHeap)[i] > (*maxHeap)[j]
})
ans := (*maxHeap)[len(*maxHeap)-1]
*maxHeap = (*maxHeap)[0 : len(*maxHeap)-1]
return ans
}
执行结果如下:

2021-07-30:两个有序数组间相加和的Topk问题。给定两个有序数组arr1和arr2,再给定一个整数k,返回来自arr1和arr2的两个数相加和最大的前k个,两个数必须分别来自两个数组。按照降的更多相关文章
- 两个数组各个数相加或相乘变成一个矩阵求第K大
input 1<=T<=20 1<=n<=100000,1<=k<=n*n a1 a2 ... an 0<ai<=10000 b1 b2 ... bn ...
- 给定两个数组,这两个数组是排序好的,让你求这两个数组合到一起之后第K大的数。
题目:给定两个数组,这两个数组是排序好的,让你求这两个数组合到一起之后第K大的数. 解题思路: 首先取得数组a的中位数a[aMid],然后在b中二分查找a[aMid],得到b[bMid],b[bSt] ...
- 在O(N)时间内求解 正数数组中 两个数相加的 最大值
一,问题描述 给定一个正数数组arr(即数组元素全是正数),找出该数组中,两个元素相加的最大值,其中被加数的下标大于加数的下标.由加法运算的可逆性,j >i 这个条件可以去掉. 即求出: max ...
- 作业帮:给定一个整数数组,找出其中两个数相加等于目标值(去重set)
题目描述 给定一个整数数组,找出其中两个数相加等于目标值 输入 [1,3,5,7,9,11] 10 输出 1,9 3,7 代码: import java.util.HashMap; import ja ...
- 两个序列求前k大和
---恢复内容开始--- 没有题目,没有题意,这是学长提过的一个技巧,给你两个排好序的序列,每次可以各从中取一个,求前k大的和, 一个优先队列,先将a序列中最大的那个和b序列所有元素相加存进队列中,每 ...
- CUDA学习(三)之使用GPU进行两个数相加
在CPU上定义两个数并赋值,然后使用GPU核函数将两个数相加并返回到CPU,在CPU上显示 #include "cuda_runtime.h" #include "dev ...
- 17.从键盘上输入一个正整数n,请按照以下五行杨辉三角形的显示方式, 输出杨辉三角形的前n行。请采用循环控制语句来实现。 (三角形腰上的数为1,其他位置的数为其上一行相邻两个数之和。) 1 1 1 1 2 1 1 3 3 1 1 4 6 4 1 1 5 10 10 5 1
17.从键盘上输入一个正整数n,请按照以下五行杨辉三角形的显示方式, 输出杨辉三角形的前n行.请采用循环控制语句来实现. (三角形腰上的数为1,其他位置的数为其上一行相邻两个数之和.) 1 1 1 1 ...
- 【LeetCode-面试算法经典-Java实现】【002-Add Two Numbers (单链表表示的两个数相加)】
[002-Add Two Numbers (单链表表示的两个数相加)] 原题 You are given two linked lists representing two non-negative ...
- 函数bsxfun,两个数组间元素逐个计算的二值操作
转自http://www.cnblogs.com/rong86/p/3559616.html 函数功能:两个数组间元素逐个计算的二值操作 使用方法:C=bsxfun(fun,A,B) 两个数组A合B间 ...
- 牛客网2016.4.11(两个数相加为sum/计数一个int型的二进制有多少个1/二叉树是否左右对称)
求最小的两个数相加为sum //求最小的两个数相加为sum public ArrayList<Integer> FindNumbersWithSum(int [] array,int su ...
随机推荐
- PYCHARM开源版-免费版本
下载地址:https://www.jetbrains.com/pycharm/download/#section=windows 亲测可以使用,不需要任何破解工具
- MAC范洪攻击-macof
macof 目的:攻击交换机的路由表,实现网络信息嗅探 macof是dsniff中的一个小工具 概要:交换机中存在着一个记录着MAC地址的表,为了完成数据的快速转发,这个表有着自动学习机制,学习后可以 ...
- 如何使用Mutex确保并发程序的正确性
1. 简介 本文的主要内容是介绍Go中Mutex并发原语.包含Mutex的基本使用,使用的注意事项以及一些实践建议. 2. 基本使用 2.1 基本定义 Mutex是Go语言中的一种同步原语,全称为Mu ...
- 【备忘录】 主定理 Master Theorem (转载)
备忘录 https://zhuanlan.zhihu.com/p/113406812
- 【微信网页授权】SpringBoot+uniapp实现网页授权获取用户基本信息
前言 缘由 起因于本狗上一个项目本打算采用微信公众号网页授权登录做用户鉴权,但最终因公众号是未认证的订阅号,无权限获取用户信息,所以改变思路,采用登录注册方式实现用户区分.但在开发中,学习了微信网页授 ...
- Linux文件上传下载--rz/sz命令
原文地址:https://www.cnblogs.com/igoodful/p/14694038.html 1.rz 命令 1.1 命令简介 rz 命令(Receive ZMODEM),使用 ZMOD ...
- RabbitMQ 消息中间件总结
RabbitMQ 是实现高级消息队列协议(AMQP:Advanced Message Queue Protocol)的开源代理软件,也称为面向消息的中间件.支持多种操作系统.多种编程语言.Rabbit ...
- 利用Karlibr生成April标定板图像
1 关键的命令 rosrun kalibr kalibr_create_target_pdf --type apriltag --nx 6 --ny 6 --tsize 0.02 --tspace 0 ...
- Little Snitch 5 - Mac 老牌防火墙安全工具软件小飞贼,监控和组织特定软件的网络连接
一旦连接到Internet,应用程序就可以随时随地发送它们想要的任何东西.通常情况下,他们是为你做的.但有时,例如在跟踪软件.木马或其他恶意软件的情况下,它们不会.但你不会注意到任何东西,因为所有这些 ...
- scp上传文件No such file or directory 问题
问题描述 window使用scp复制文件到linux时报 No such file or directory 错误 解决办法 本地复制到Linux,win环境必须用绝对路径 上传成功