【Leetcode堆和双端队列】滑动窗口最大值(239)
题目
给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。
返回滑动窗口中的最大值。
示例:
输入: nums = [1,3,-1,-3,5,3,6,7], 和 k = 3
输出: [3,3,5,5,6,7]
解释:
滑动窗口的位置 最大值
--------------- -----
[1 3 -1] -3 5 3 6 7 3
1 [3 -1 -3] 5 3 6 7 3
1 3 [-1 -3 5] 3 6 7 5
1 3 -1 [-3 5 3] 6 7 5
1 3 -1 -3 [5 3 6] 7 6
1 3 -1 -3 5 [3 6 7] 7
提示:
你可以假设 k 总是有效的,在输入数组不为空的情况下,1 ≤ k ≤ 输入数组的大小。
进阶:
你能在线性时间复杂度内解决此题吗?
解答
方法一:暴力破解。cur表示窗口,记录最大值,依次后移。时间复杂度O(n·k),太慢了,才超过30%的代码
方法二:用大顶堆存窗口元素,堆顶就是当前窗口最大值,依次后移,删除最左边,加入一个新元素,堆删除操作时间复杂度O(log(n)),插入操作O(1),这种方法总的时间复杂度O(n·log(k))。
But!Python中没有大顶堆的实现,用“存入堆、从堆中取出的时候,都用相反数”模拟大顶堆可行,以为这就解决了吗? No! heapq包可以删除堆顶元素,但没提供删除堆中某元素的方法,所有先找到元素在堆中位置再用了del删除,del删除后,存储堆的列表顺序就不是堆了,需要重新排列成堆,时间复杂度O(n),所以用Python最后做出来时间复杂度O(n·k),但这个提交速度为毛没有暴力破解快,无语。。。
方法三:双向队列,时间复杂度O(n)
思路:依次遍历列表中的元素,cur存放已经在窗口中的元素的下标,假如当前元素为c, 则用c和cur中元素从后向前比较,删除cur中小于c的元素,并把c的下标加到cur中,这步操作保证了cur[0]表示的元素永远是窗口中最大的
注:c和cur比较时,一定要从后往前,否则部分样例不能通过:[1, 3, 1, 2, 0, 5], k=3 .........................视频里讲的从前往后:)
通过代码如下:
from heapq import *
class Solution:
# 方法一:暴力破解
# def maxSlidingWindow(self, nums: List[int], k: int) -> List[int]:
# if not nums:
# return []
# ans = []
# cur = []
# for x in range(k):
# cur.append(nums[x])
# ans.append(max(cur))
# while x < len(nums)-1:
# del cur[0] # 删除表头
# cur.append(nums[x+1]) # 添加到尾
# ans.append(max(cur)) # max()的平均时间复杂度是O(n)
# x = x+1
# return ans
# 方法二:大顶堆
# def maxSlidingWindow(self, nums: List[int], k: int) -> List[int]:
# if not nums:
# return []
# l = [] # 预存堆
# ans = []
# for index, x in enumerate(nums):
# if index==0 or len(l)<k: # 堆还没满,入堆
# heappush(l, -x)
# else:
# ans.append(-l[0]) # 堆顶,最小的,取负数刚好是最大的
# del l[l.index(-nums[index-k])] # 删除窗口左侧第一个
# heappush(l, -x) # 入堆
# heapify(l) # 堆貌似乱了......, 重新排列成堆,时间复杂度是O(n)。。。
# ans.append(-l[0])
# return ans
# 方法三:双端队列
def maxSlidingWindow(self, nums, k):
if not nums:
return []
cur = [] # 存窗口元素的index,cur[0]位置永远是最大的
ans = []
for index, x in enumerate(nums):
if index >= k and index - k >= cur[0]: # cur[0]和当前元素的index距离为k,说明cur[0]该从窗口删除了
cur.pop(0)
while cur and x >= nums[cur[-1]]: # 这里,cur要从后往前删小于当前元素的值
cur.pop()
cur.append(index)
if index + 1 >= k: # 当前元素位已经走过了窗口距离大小
ans.append(nums[cur[0]])
return ans
【Leetcode堆和双端队列】滑动窗口最大值(239)的更多相关文章
- python基础教程_学习笔记19:标准库:一些最爱——集合、堆和双端队列
版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/signjing/article/details/36201499 标准库:一些最爱 集合.堆和双端队 ...
- lintcode 滑动窗口的最大值(双端队列)
题目链接:http://www.lintcode.com/zh-cn/problem/sliding-window-maximum/# 滑动窗口的最大值 给出一个可能包含重复的整数数组,和一个大小为 ...
- [Leetcode]双项队列解决滑动窗口最大值难题
这道题是从优先队列的难题里面找到的一个题目.可是解法并不是优先队列,而是双项队列deque 其实只要知道思路,这一道题直接写没有太大的问题.我们看看题 给定一个数组 nums,有一个大小为 k 的滑动 ...
- 关于双端队列 deque 模板 && 滑动窗口 (自出)
嗯... deque 即为双端队列,是c++语言中STL库中提供的一个东西,其功能比队列更强大,可以从队列的头与尾进行操作... 但是它的操作与队列十分相似,详见代码1: 1 #include < ...
- 剑指 Offer 59 - I. 滑动窗口的最大值 + 双指针 + 双端队列
剑指 Offer 59 - I. 滑动窗口的最大值 Offer_59_1 题目详情 方法一:暴力方法+双指针 package com.walegarrett.offer; /** * @Author ...
- C#LeetCode刷题之#641-设计循环双端队列(Design Circular Deque)
问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/4132 访问. 设计实现双端队列. 你的实现需要支持以下操作: M ...
- Java实现 LeetCode 641 设计循环双端队列(暴力)
641. 设计循环双端队列 设计实现双端队列. 你的实现需要支持以下操作: MyCircularDeque(k):构造函数,双端队列的大小为k. insertFront():将一个元素添加到双端队列头 ...
- LeetCode - 滑动窗口最大值
给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧.你只可以看到在滑动窗口内的 k 个数字.滑动窗口每次只向右移动一位. 返回滑动窗口中的最大值. 输入: nums ...
- Leetcode 239题 滑动窗口最大值(Sliding Window Maximum) Java语言求解
题目链接 https://leetcode-cn.com/problems/sliding-window-maximum/ 题目内容 给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧 ...
随机推荐
- [Day6] Nginx 进阶模块
一. 使用变量防盗链referer模块 功能:通过验证referer请求头是否合法,来拒绝非正常的网站访问我们站点的资源 思路:通过referer模块,用invaild_refereri变量根据配置判 ...
- iOS CALayer之CAEmitterLayer粒子发射器的神奇效果
https://www.jianshu.com/p/c54ffd7412e7 想必以前QQ空间的点赞效果大家都知道吧,点赞之后按钮周围会有一圈爆裂的小圆点:还有微信的红包雨表情动画等,以及烟花,火焰效 ...
- vue项目中实现扫码功能
项目地址:https://github.com/wkl007/vue-scan-demo.git 项目主要是做的一个扫码的功能 核心代码为 <div class="scan" ...
- curl下载安装与使用
下载: https://curl.haxx.se/download.html 安装: 二进制安装.即解压即可. 使用 1.获取页面内容. 不加任何参数时,默认会发送GET请求来获取url内容到标准输出 ...
- leyou_01_环境搭建
1.乐优商城项目搭建 前端技术: 基础的HTML.CSS.JavaScript(基于ES6标准) JQuery Vue.js 2.0以及基于Vue的框架:Vuetify 前端构建工具:WebPack ...
- TZ_16_Vue_入门案例
1.新建一个html文件导入vue.js <script src="node_modules/vue/dist/vue.js"></script> 2.创建 ...
- step()动画
<style type="text/css"> .hi { width: 50px; height: 72px; background-image: url(" ...
- shell linux基本命令实例、笔记
1. 在当前文件夹下.查找20分钟内,被訪问过的文件, 并将文件的详情显示出来: find ./ -name '*.log' -mmin -20 -exec ls -l {} \; 当然,须要指出 ...
- 在Linux系统下进入MySql数据库进行操作
例: ---- 1.进入mysql数据库 root@test:/home# mysql -uroot -proot <uroot是用户名,proot是密码> 2.查询所有的库 my ...
- day37 03-Hibernate二级缓存:集合缓冲区特点
所以说要经常检查hibernate3的核心配置文件hibernate.cfg.xml. Hibernate: select customer0_.cid as cid0_0_, customer0_. ...