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 ...
随机推荐
- ADOStoredProc动态调用存储过程
有些用ADOStoredProc进行操作时,要动态创建输入输出参数,一定要用 ADOStoredProc1.Parameters.Refresh;将参数进行刷新一下.
- Java 面试手撕代码
1. 判断括号有效性 public static boolean fun5(String str) { HashMap<Character, Character> hashMap = ne ...
- SpringBoot笔记--文件配置加载顺序+整合其他框架
内部文件配置加载顺序 外部文件配置加载顺序 jar包配置 整合Junit 若是业务管理类和测试类在同一个包下面,那么这句话, 可以不加括号,只写注解名称 否则,就必须指定到包下面,不然会报错 整合Re ...
- IO 流分类
更多内容,前往 IT-BLOG 一.File File 类(磁盘操作)可以用于表示文件和目录的信息,但是它不表示文件的内容.递归地列出一个目录下所有文件: 1 public static void l ...
- 主板芯片组驱动和Win系统版本互相关联
主板芯片组驱动和Win系统版本互相关联,过早的系统安装较新版的芯片组驱动,或者较新版本的操作系统安装旧版的芯片组驱动,都可能导致系统不稳定蓝屏.解决方案就是安装最新的芯片组驱动和最新版的操作系统.
- 剑指 offer 第 2 天
第 2 天 链表(简单) 剑指 Offer 06. 从尾到头打印链表 输入一个链表的头节点,从尾到头反过来返回每个节点的值(用数组返回). 示例 1: 输入:head = [1,3,2] 输出:[2, ...
- flutter---->flutter orientation
Get the orientation 1. Media Query import 'package:flutter/material.dart'; void main() => runApp( ...
- 超详细!新手如何创建一个Vue项目
目录 一.在官网下载Vue.js 二.使用<script>标签直接引入本地的vue.js 三.使用CDN引入Vue.js 四.验证是否安装成功 五.安装Vue Devtools浏览器调试插 ...
- ChatGPT推荐最常用的自动化测试、性能、安全测试工具!
ChatGPT是一种当前被广泛关注的人工智能技术,它具备生成自然语言的能力,能够完成一些简单的文本生成.对话交互等任务.ChatGPT 算法的出现,打破了以前自然语言处理的瓶颈,使得机器具备了更加贴合 ...
- Redis 数据类型 Set
Redis 数据类型 Set(集合) Redis 常用命令,思维导图 >>> Redis 的 Set 是 String 类型的无序集合.集合成员是唯一的,这就意味着集合中不能出现重复 ...