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
}

执行结果如下:


左神java代码

2021-07-30:两个有序数组间相加和的Topk问题。给定两个有序数组arr1和arr2,再给定一个整数k,返回来自arr1和arr2的两个数相加和最大的前k个,两个数必须分别来自两个数组。按照降的更多相关文章

  1. 两个数组各个数相加或相乘变成一个矩阵求第K大

    input 1<=T<=20 1<=n<=100000,1<=k<=n*n a1 a2 ... an 0<ai<=10000 b1 b2 ... bn ...

  2. 给定两个数组,这两个数组是排序好的,让你求这两个数组合到一起之后第K大的数。

    题目:给定两个数组,这两个数组是排序好的,让你求这两个数组合到一起之后第K大的数. 解题思路: 首先取得数组a的中位数a[aMid],然后在b中二分查找a[aMid],得到b[bMid],b[bSt] ...

  3. 在O(N)时间内求解 正数数组中 两个数相加的 最大值

    一,问题描述 给定一个正数数组arr(即数组元素全是正数),找出该数组中,两个元素相加的最大值,其中被加数的下标大于加数的下标.由加法运算的可逆性,j >i 这个条件可以去掉. 即求出: max ...

  4. 作业帮:给定一个整数数组,找出其中两个数相加等于目标值(去重set)

    题目描述 给定一个整数数组,找出其中两个数相加等于目标值 输入 [1,3,5,7,9,11] 10 输出 1,9 3,7 代码: import java.util.HashMap; import ja ...

  5. 两个序列求前k大和

    ---恢复内容开始--- 没有题目,没有题意,这是学长提过的一个技巧,给你两个排好序的序列,每次可以各从中取一个,求前k大的和, 一个优先队列,先将a序列中最大的那个和b序列所有元素相加存进队列中,每 ...

  6. CUDA学习(三)之使用GPU进行两个数相加

    在CPU上定义两个数并赋值,然后使用GPU核函数将两个数相加并返回到CPU,在CPU上显示 #include "cuda_runtime.h" #include "dev ...

  7. 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 ...

  8. 【LeetCode-面试算法经典-Java实现】【002-Add Two Numbers (单链表表示的两个数相加)】

    [002-Add Two Numbers (单链表表示的两个数相加)] 原题 You are given two linked lists representing two non-negative ...

  9. 函数bsxfun,两个数组间元素逐个计算的二值操作

    转自http://www.cnblogs.com/rong86/p/3559616.html 函数功能:两个数组间元素逐个计算的二值操作 使用方法:C=bsxfun(fun,A,B) 两个数组A合B间 ...

  10. 牛客网2016.4.11(两个数相加为sum/计数一个int型的二进制有多少个1/二叉树是否左右对称)

    求最小的两个数相加为sum //求最小的两个数相加为sum public ArrayList<Integer> FindNumbersWithSum(int [] array,int su ...

随机推荐

  1. python调用adb shell

    最近在用python做一个小工具,自动执行一些adb shell命令,使用subprocess.Popen来实现. 不过遇到个问题就是执行adb shell后就无法执行后面adb shell里的命令了 ...

  2. C#比较类/接口、Dictionary 排序

    作者:l625208058 链接:https://www.jianshu.com/p/cd1be6652570 先 F12 看下 List.Sort() 方法 public void Sort(int ...

  3. sdut——4541:小志志和小峰峰的日常(取石子博弈模板题 4合1)

    小志志和小峰峰的日常 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 小志志和小峰峰特别喜欢一起讨论一些很好玩的问题.  ...

  4. mac快捷键和win10快捷键和mma快捷手册

    不定期更新 来自知乎,b站等 mac下的快捷键 如果你mac接了个不一致的键盘,mac会让你检测,(按左ctrl右边的键,按右ctrl左边的键),之后会进行键位映射,这也太复杂了,我拒绝记录. com ...

  5. 如何获取obs视频帧的二进制数据

    前面几篇文章梳理了obs的录屏和推流流程,几条纵线整理下来,算是基本理清了obs的工作流程. 现在回到第一个目标:捕捉桌面的帧数据,用rendertarget显示并输出到UE5材质. 那么,帧数据到底 ...

  6. java面向对象-基础入门

    java面向对象-基础入门 面向过程:线性思维 面向对象思维:物以类聚,分类的思维 对于描述复杂的事物,为了从宏观上把握,从整体上合理分析,我们需要使用面向对象的思路来分析整个系统,但是具体到某个微观 ...

  7. Nacos 实现 AP+CP原理[Raft 算法 NO]

    来源于网络 一.什么是 Raft算法 Raft 适用于一个管理日志一致性的协议,相比于 Paxos 协议 Raft 更易于理解和去实现它.为了提高理解性,Raft 将一致性算法分为了几个部分,包括领导 ...

  8. 100 多个常用免费 API 接口推荐与分享,收藏备用

    写在最前 各类免费 API 接口整理,主要是 APISpace 上和其他各类开放平台上的一些,有需要的赶紧收藏备用.   高德地图 标准图层 TileLayer 卫星图层 TileLayer.Sate ...

  9. window身上的方法 弹出框/打开和关闭

    window身上的方法内置函数 alert() parseInt() parseFloat() setInterval(); setTimeout(); clearTimeout(); clearIn ...

  10. nodejs,,一些基本操作--server。js

    1.解决中文乱码问题: const http = require('http') const server = http.createServer((req, res) => { // 设置字符 ...