LintCode领扣 题解丨字节跳动面试题:第k大的子数组

简介: LintCode领扣 题解丨字节跳动面试题:第k大的子数组

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

1≤n≤10​^5
1≤a​i≤10^​9
1≤k≤​n(n+1)/2
在线评测地址:https://www.lintcode.com/problem/the-kth-subarray/?utm_source=sc-tc-sz0810)

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) {
    long tmp1 = 0, tmp2 = 0, now = a[0];
    int l = -1, r = 0, n = a.length;
    long all = (long)n * (n + 1) / 2;
    while (l <= r && r < n)
    {
        if (now >= x) {
            if (now == x) {
                tmp2++;
            } else {
                tmp1++;
            }
            tmp1 += n - r - 1;
            l++; now -= a[l];
        }
        else {r++; if (r < n) now += a[r];}
    }
    if (all - tmp1 - tmp2 < k && all - tmp1 >= k) return 0;
    if (all - tmp1 - tmp2 >= k) return 1;
    else return -1;
} 
public long thekthSubarray(int[] a, long k) {
    // wrrite your code here
    int n = a.length;
    long sum = 0;
    for (int i = 0; i < n; i++) {
        sum += a[i];
    }
    long l = 1, r = sum;
    while (l <= r) {
        long mid = (l + r) / 2;
        int flag = check(mid, a, k);
        if (flag == 0) {
            return mid;
        }
        if (flag == 1) {
            r = mid - 1;
        }
        else {
            l = mid + 1;
        }
    }
    return -1;
}

}
更多题解参见:[九章算法官网]https://www.jiuzhang.com/solution/the-kth-subarray/?utm_source=sc-tc-sz0810

相关文章
|
4月前
|
存储 Unix 程序员
面试题:Ctrl + C在不同操作系统下的应用
字节跳动面试题:Ctrl + C在不同操作系统下的应用
88 1
|
4月前
|
负载均衡 算法 应用服务中间件
面试题:Nginx有哪些负载均衡算法?Nginx位于七层网络结构中的哪一层?
字节跳动面试题:Nginx有哪些负载均衡算法?Nginx位于七层网络结构中的哪一层?
107 0
|
4月前
|
设计模式 缓存 Dart
Flutter学习笔记&学习资料推荐,15分钟的字节跳动视频面试
Flutter学习笔记&学习资料推荐,15分钟的字节跳动视频面试
|
4月前
|
Python 计算机视觉
2024年Python最新利用python进行数学公式识别_python 识别图片中的数学公式,2024年最新字节跳动技术岗位面试
2024年Python最新利用python进行数学公式识别_python 识别图片中的数学公式,2024年最新字节跳动技术岗位面试
2024年Python最新利用python进行数学公式识别_python 识别图片中的数学公式,2024年最新字节跳动技术岗位面试
|
4月前
|
Oracle 关系型数据库 数据库
Oracle 部署及基础使用,字节跳动资深面试官亲述
Oracle 部署及基础使用,字节跳动资深面试官亲述
|
4月前
|
机器学习/深度学习 数据采集 算法
2024年机器学习入门,2024年最新字节跳动视频面试一般多久会收到结果
2024年机器学习入门,2024年最新字节跳动视频面试一般多久会收到结果
2024年机器学习入门,2024年最新字节跳动视频面试一般多久会收到结果
|
4月前
|
IDE 开发工具 开发者
2024年最新5个提升生产效率的Python开发和配置的小技巧_python高级开发技巧,字节跳动面试必问
2024年最新5个提升生产效率的Python开发和配置的小技巧_python高级开发技巧,字节跳动面试必问
2024年最新5个提升生产效率的Python开发和配置的小技巧_python高级开发技巧,字节跳动面试必问
|
4月前
|
前端开发 程序员 开发工具
2024年最全0基础程序员如何快速进阶成为编程老司机?_码农速成(2),字节跳动面试攻略
2024年最全0基础程序员如何快速进阶成为编程老司机?_码农速成(2),字节跳动面试攻略
2024年最全0基础程序员如何快速进阶成为编程老司机?_码农速成(2),字节跳动面试攻略
|
4月前
|
机器学习/深度学习 数据采集 自然语言处理
2024年Python最新【python开发】并发编程(下),2024年最新字节跳动的面试流程
2024年Python最新【python开发】并发编程(下),2024年最新字节跳动的面试流程
2024年Python最新【python开发】并发编程(下),2024年最新字节跳动的面试流程
|
4月前
|
程序员 开发工具 Python
最全学Python有什么用?看完这些你肯定明白_学pysion的作用,2024年最新字节跳动面试严格吗
最全学Python有什么用?看完这些你肯定明白_学pysion的作用,2024年最新字节跳动面试严格吗
最全学Python有什么用?看完这些你肯定明白_学pysion的作用,2024年最新字节跳动面试严格吗