大厂面试真题详解:滑动窗口的中位数

简介: 大厂面试真题详解:滑动窗口的中位数

给定一个包含 n 个整数的数组,和一个大小为 k 的滑动窗口,从左到右在数组中滑动这个窗口,找到数组中每个窗口内的中位数。(如果数组个数是偶数,则在该窗口排序数字后,返回第 N/2 个数字。)

在线评测地址:领扣题库官网

样例 1:

输入:
[1,2,7,8,5]
3
输出:
[2,7,7]

解释:
最初,窗口的数组是这样的:[ | 1,2,7 | ,8,5] , 返回中位数 2;
接着,窗口继续向前滑动一次。[1, | 2,7,8 | ,5], 返回中位数 7;
接着,窗口继续向前滑动一次。[1,2, | 7,8,5 | ], 返回中位数 7;

样例 2:

输入:
[1,2,3,4,5,6,7]
4
输出:
[2,3,4,5]

解释:
最初,窗口数组是这样的:[ | 1,2,3,4, | 5,6,7] , 返回中位数 2;
接着,窗口向前滑动一次.[1,| 2,3,4,5 | 6,7],返回中位数 3;
接着,窗口向前滑动一次.[1,2, | 3,4,5,6 | 7 ], 返回中位数 4;
接着,窗口向前滑动一次[1,2,3,| 4,5,6,7 ], 返回中位数 5;

题解:
使用九章算法强化班中讲到的 HashHeap。即一个 Hash + Heap。 Hash 的 key 是 Heap 里的每个元素,值是这个元素在 Heap 中的下标。

要做这个题首先需要先做一下 Data Stream Median。这个题是只在一个集合中增加数,不删除数,然后不断的求中点。 Sliding Window Median,就是不断的增加数,删除数,然后求中点。比 Data Stream Median 难的地方就在于如何支持删除数。

因为 Data Stream Median 的方法是用 两个 Heap,一个 max heap,一个min heap。所以删除的话,就需要让 heap 也支持删除操作。 由于 Python 的 heapq 并不支持 logn 时间内的删除操作,因此只能自己实现一个 hash + heap 的方法。

总体时间复杂度 O(nlogk),n是元素个数,k 是 window 的大小。

    public class Solution {
        /*
         * @param nums: A list of integers
         * @param k: An integer
         * @return: The median of the element inside the window at each moving
         */
        PriorityQueue<Integer> maxHeap, minHeap;
        public List<Integer> medianSlidingWindow(int[] A, int k) {
            // write your code here
            List<Integer> res = new ArrayList<Integer>();
            int n = A.length;
            if (n == 0) {
                return res;
            }
            
            maxHeap = new PriorityQueue<Integer>(n, Collections.reverseOrder());
            minHeap = new PriorityQueue<Integer>(n);
            
            int i;
            for (i = 0; i < n; ++i) {
                if (maxHeap.size() == 0 || A[i] <= maxHeap.peek()) {
                    maxHeap.offer(A[i]);
                }
                else {
                    minHeap.offer(A[i]);
                }
                
                balance();
                if (i - k >= 0) {
                    if (A[i - k] > maxHeap.peek()) {
                        minHeap.remove(A[i - k]);
                    }
                    else {
                        maxHeap.remove(A[i - k]);
                    }
                }
                
                balance();
                
                if (i >= k - 1) {
                    res.add(maxHeap.peek());
                }
            }
            
            return res;
        }
        
        private void balance() {
            while (maxHeap.size() < minHeap.size()) {
                maxHeap.offer(minHeap.poll());
            }
            
            while (minHeap.size() < maxHeap.size() - 1) {
                minHeap.offer(maxHeap.poll());
            }
        }
    }

更多题解参考:九章官网solution

相关文章
|
3月前
|
SQL 数据挖掘 数据处理
「SQL面试题库」 No_20 给定数字的频率查询中位数
「SQL面试题库」 No_20 给定数字的频率查询中位数
|
3月前
|
SQL 数据挖掘 数据处理
「SQL面试题库」 No_1 员工薪水中位数
「SQL面试题库」 No_1 员工薪水中位数
|
4月前
剑指Offer LeetCode 面试题59 - I. 滑动窗口的最大
剑指Offer LeetCode 面试题59 - I. 滑动窗口的最大
21 0
|
9月前
【牛客面试必刷TOP101】有效括号序列、滑动窗口的最大值
【牛客面试必刷TOP101】有效括号序列、滑动窗口的最大值
|
安全 测试技术 Python
软件测试面试题及答案,这个题库有3千多道最新面试真题可以刷
相信对于很多软件测试新手来说,技术项目的面试是十分让人头疼的,生怕没回答得好,就会跟这个offer失之交臂
149 0
|
Serverless C语言 索引
华为面试C语言真题(二)
华为面试C语言真题( 二)
263 1
华为面试C语言真题(二)
|
设计模式 缓存 算法
腾讯Java高级岗180道面试真题,面试大厂拿45Koffer没问题!
一、数据结构与算法基础 · 说一下几种常见的排序算法和分别的复杂度。 · 用Java写一个冒泡排序算法 · 描述一下链式存储结构。 · 如何遍历一棵二叉树? · 倒排一个LinkedList。 · 用Java写一个递归遍历目录下面的所有文件
|
算法 数据库 UED
面试常见问题-限流策略有哪些,滑动窗口算法和令牌桶区别,使用场景
面试常见问题-限流策略有哪些,滑动窗口算法和令牌桶区别,使用场景
541 0
|
消息中间件 NoSQL 网络协议
字节跳动三面Java经历,砍下年薪50W的Offer,面试真题整理分享
应广大读者要求,今天开更一些大厂的面经和相关的面试干货,下面这份**最新字节跳动春招面经+笔记**带给大家。
119 0
字节跳动三面Java经历,砍下年薪50W的Offer,面试真题整理分享