POJ 2823 Sliding Window 滑动窗口 单调队列 Monotone Queue

Sliding Window

Time Limit: 12000MS Memory Limit: 65536K
Total Submissions: 65974 Accepted: 18744
Case Time Limit: 5000MS

Description

An array of size n ≤ 10 6 is given to you. There is a sliding window of size k which is moving from the very left of the array to the very right. You can only see the k numbers in the window. Each time the sliding window moves rightwards by one position. Following is an example:
The array is [1 3 -1 -3 5 3 6 7], and k is 3.
Window positionMinimum valueMaximum value
[1  3  -1] -3  5  3  6  7 -13
 1 [3  -1  -3] 5  3  6  7 -33
 1  3 [-1  -3  5] 3  6  7 -35
 1  3  -1 [-3  5  3] 6  7 -35
 1  3  -1  -3 [5  3  6] 7 36
 1  3  -1  -3  5 [3  6  7]37

Your task is to determine the maximum and minimum values in the sliding window at each position.

Input

The input consists of two lines. The first line contains two integers n and k which are the lengths of the array and the sliding window. There are n integers in the second line.

Output

There are two lines in the output. The first line gives the minimum values in the window at each position, from left to right, respectively. The second line gives the maximum values.

Sample Input

8 3
1 3 -1 -3 5 3 6 7

Sample Output

-1 -3 -3 -3 3 3
3 3 5 5 6 7

Source

 

分析

这道题是一个典型的单调队列例题。我们在这题当中有两个小问题,分别求最大最小。我们不急着一次完成,分两个步骤更为妥当。

我们要做的是维护一个单调队列,若求最大,那么维护一个单调递减的队列,反之亦然。

单调队列在这道题中的简单思路,我们以求最大来距离:

入队一个数,

从后往前删,直到队尾的数大于入队数或队列为空。这里的原理是,当前窗口出现了一个大于之前的数字,那么之后的窗口当中,怎么也不可能再用到那些比它小的数字了,从队尾删。也就是代码13~14行。

我们再从队头找,如果找到的数已经到窗口外面去了,那么就直接删除。这里的原理是,我们维护的这个单调队列是的队列里越靠前的数,越处于数列的前部。道理很简单,我们入队一个数的时候,一定是的它处于队尾。也就是代码17~18行。

最后这个单调队列当中,所有的数都再窗口内,且从大到小排序。输出队头。

程序

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 int n, k, Q[1000000 + 5], a[1000000 + 5], p[1000000 + 5];
 4 int main()
 5 {
 6     cin >> n >> k;
 7     for (int i = 1; i <= n; i++)
 8         cin >> a[i];
 9     int Head = 1;
10     int Tail = 0;
11     for (int i = 1; i <= n; i++)
12     {
13         while(Head <= Tail && Q[Tail] >= a[i])
14             Tail--;
15         Q[++Tail] = a[i];
16         p[Tail]=i;
17         while (p[Head] <= i-k)
18             Head++;
19         if(i >= k)
20             cout << Q[Head] << " ";
21     }
22     cout << endl;
23     Head = 1;
24     Tail = 0;
25     for (int i = 1; i <= n; i++)
26     {
27         while (Head <= Tail && Q[Tail] <= a[i])
28             Tail--;
29         Q[++Tail] = a[i];
30         p[Tail] = i;
31         while (p[Head] <= i-k)
32             Head++;
33         if(i >= k)
34             cout << Q[Head] << " ";
35     }
36     cout << endl;
37     return 0;
38 }

 

转载于:https://www.cnblogs.com/OIerPrime/p/8546637.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我! 毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip毕设新项目-基于Java开发的智慧养老院信息管理系统源码+数据库(含vue前端源码).zip
综合小区管理系统管理系统按照操作主体分为管理员和用户。管理员的功能包括报修管理、车位管理、车位分配管理、出入管理、字典管理、房屋管理、物业费缴纳管理、公告管理、物业人员投诉管理、我的私信管理、物业人员管理、用户管理、管理员管理。用户的功能包括管理部门以及部门岗位信息,管理招聘信息,培训信息,薪资信息等。该系统采用了Mysql数据库,Java语言,Spring Boot框架等技术进行编程实现。 综合小区管理系统管理系统可以提高综合小区管理系统信息管理问题的解决效率,优化综合小区管理系统信息处理流程,保证综合小区管理系统信息数据的安全,它是一个非常可靠,非常安全的应用程序。 管理员权限操作的功能包括管理公告,管理综合小区管理系统信息,包括出入管理,报修管理,报修管理,物业费缴纳等,可以管理操作员。 出入管理界面,管理员在出入管理界面中可以对界面中显示,可以对招聘信息的招聘状态进行查看,可以添加新的招聘信息等。报修管理界面,管理员在报修管理界面中查看奖罚种类信息,奖罚描述信息,新增奖惩信息等。车位管理界面,管理员在车位管理界面中新增。公告管理界面,管理员在公告管理界面查看公告的工作状态,可以对公告的数据进行导出,可以添加新公告的信息,可以编辑公告信息,删除公告信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值