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. Less-3 和 Less-4 ')闭合绕过

    判断注入类型 测试:http://localhost/sqli-labs-master/Less-3/index.php?id=1a 正常回显,可以判断为 字符型注入 闭合字符串执行而已 SQL语句 ...

  2. GO语言学习笔记-反射篇 Study for Go ! Chapter nine - Reflect

    持续更新 Go 语言学习进度中 ...... GO语言学习笔记-类型篇 Study for Go! Chapter one - Type - slowlydance2me - 博客园 (cnblogs ...

  3. 解读 Servlet 源码:GenericServlet,ServletConfig,ServletContext

    解读 Servlet 源码:GenericServlet,ServletConfig,ServletContext 每博一文案 人活着,就得随时准备经受磨难.他已经看过一些书,知道不论是普通人还是了不 ...

  4. Jira使用浅谈篇二

    本篇参考:https://university.atlassian.com/student/collection/850385/path/1083901 本篇接上文,上文已经对项目设置了一个基础的配置 ...

  5. 面试突击:MVCC 和间隙锁有什么区别?

    MVCC 和间隙锁是两种完全不同的机制,但它们的目的都是相同的,都是用来保证数据库并发访问的,我们先来看二者的定义. MVCC 定义 MVCC 是多版本并发控制(Multi-Version Concu ...

  6. 安全测试之探索 windows 游戏扫雷

    作者:京东工业 宛煜昕 扫雷游戏相信很多人都从小玩过,在那个电脑游戏并不多的时代,扫雷成为玩的热度蛮高的一款游戏之一,然而就在有一次,接触到了一次不寻常的扫雷过程,使得后来我也有了这个冲动,也来做一次 ...

  7. $\mathcal{Mathicの代码风格}$

    概述 \(#include\) 语句必须置于整个程序的开头. 不应 using namespace foo; 若有必要可以 using foo::bar; 单行字符数必须不超过\(80\). 预编译 ...

  8. fastjson很好,但不适合我

    记者:大爷您有什么特长呀? fastjson:我很快. 记者:23423乘以4534等于多少? fastjson:等于2343. 记者:?? fastjson:你就说快不快吧! 这个略显马丽苏的标题, ...

  9. Django笔记十四之统计总数、最新纪录和空值判断等功能

    本篇笔记将介绍一些 Django 查询中统计总数.最新纪录和空值判断等功能. count in_bulk latest.earliest first.last exists contains.icon ...

  10. linux 安装 node 和 npm 服务

    1.安装文件下载 下载地址:https://nodejs.org/zh-cn/download/ 2.安装步骤 1.将安装包上传到指定位置(我习惯放到:/usr/local/application/目 ...