九章算法 | 字节跳动面试题:第k大的子数组

该博客讨论了如何解决字节跳动面试中的一道问题,即在长度为n的数组中找到第k大的子数组之和。由于子数组数量巨大,不能直接枚举排序,作者提出了利用二分法结合双指针在O(nlog(n))时间内求解的方法。通过二分查找确定目标和,再用双指针计算小于目标和的子数组数量,从而找到第k大的和。
摘要由CSDN通过智能技术生成

给定一个长度为n的数组a,它有n(n+1)/2​​个子数组。请计算这些子数组的和,然后按照升序排列,并返回排序后第k个数。

  • 1≤n≤10​^5
  • 1≤ai≤10^​9
  • 1≤k≤​n(n+1)/2

在线评测地址:LintCode 领扣

Example1

Input: 
[2,3,1,4]
6
Output:5
Explanation:
我们可以得到所有子数组的和是 [1,2,3,4,4(3 + 1), 5(1 + 4), 5(2 + 3), 6(2 + 3 + 1), 8(3 + 1 + 4), 10(2 + 3 + 1 + 4)]。其中第六个是5。

【题解】

算法

二分+two pointer

算法分析

我们可以看到,题目需要求和第k

k大的子区间,而我们的区间总个数共有n(n+1)/2个,当n为10^5​​时这个数高达10^10级别。我们显然不能暴力的枚举每一个区间和然后排序。

算法思路

我们注意到,所有数字的和不超过10^14,这个范围可以让我们想到使用二分最终的答案进行求解。

二分要求解的问题是:对于给定的和x,求有多少个区间的和小于x,小于等于x。这需要我们在O(n)的时间复杂度内进行求解。由于数组内所有数都是正数,我们自然的可以想到同向双指针求解。当当前区间的和大于k,就移动左指针,否则移动右指针。

时间复杂度

O(nlog(n))

public class Solution {
     
    /**
     * @param a: an array
     * @param k: the kth
     * @return: return the kth subarray
     */
     
    private int check(long x, int[] a, long k)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值