题目大意

给定一行数,共N个。有一个长度为K的窗口从左向右滑动,窗口中始终有K个数字,窗口每次滑动一个数字。求各个时刻窗口中的最大值和最小值。

题目分析

直接搜索,复杂度为O(n^2)。本题可以看做是一个区间求最大值最小值的问题,因此考虑使用线段树解决。 
    和用单调队列方法相比,时间复杂度O(nlogn),略慢。但也是一个不错的算法,毕竟进行区间求值,单调队列并不总是通用。

实现(c++)

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<vector>
using namespace std;
#define MAX_SIZE 1000005
#define INFINITE 1 << 30
#define MAX(a, b) a > b?a:b
#define MIN(a, b) a < b?a:b
int gResult[MAX_SIZE][2]; struct Node{
int beg;
int end;
int max;
int min;
};
Node gNodes[MAX_SIZE * 4];
int gArray[MAX_SIZE];
void PushUp(int index){
int left = (index << 1) + 1;
int right = (index << 1) + 2;
gNodes[index].max = MAX(gNodes[left].max, gNodes[right].max);
gNodes[index].min = MIN(gNodes[left].min, gNodes[right].min);
} void BuildTree(int index, int beg, int end){
gNodes[index].beg = beg;
gNodes[index].end = end;
if (beg == end){
gNodes[index].min = gNodes[index].max = gArray[beg];
return;
}
int left = (index << 1) + 1;
int right = (index << 1) + 2;
int mid = (beg + end) >> 1;
BuildTree(left, beg, mid);
BuildTree(right, mid + 1, end);
PushUp(index);
} pair<int,int> Query(int index, int beg, int end){ if (beg > end || beg > gNodes[index].end || end < gNodes[index].beg){
return pair<int, int>(-INFINITE, INFINITE);
}
if (gNodes[index].beg >= beg && gNodes[index].end <= end){
return pair<int, int>(gNodes[index].max, gNodes[index].min);
}
int mid = (gNodes[index].beg + gNodes[index].end) >> 1;
int left = (index << 1) + 1;
int right = (index << 1) + 2;
pair<int, int> result_left = Query(left, beg, MIN(mid, end));
pair<int, int> result_right = Query(right, MAX(mid + 1, beg), end);
return pair<int, int>(MAX(result_left.first, result_right.first), MIN(result_left.second, result_right.second));
}
int main(){
int n, k;
scanf("%d %d", &n, &k);
for (int i = 0; i < n; i++){
scanf("%d", gArray + i);
}
BuildTree(0, 0, n - 1);
pair<int, int> result;
for (int i = 0; i <= n - k; i++){
result = Query(0, i, i + k-1);
gResult[i][0] = result.first;
gResult[i][1] = result.second;
}
for (int i = 0; i <= n - k; i++){
printf("%d ", gResult[i][1]);
}
printf("\n");
for (int i = 0; i <= n - k; i++){
printf("%d ", gResult[i][0]);
}
printf("\n");
return 0;
}

poj_2823 线段树的更多相关文章

  1. bzoj3932--可持久化线段树

    题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...

  2. codevs 1082 线段树练习 3(区间维护)

    codevs 1082 线段树练习 3  时间限制: 3 s  空间限制: 128000 KB  题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...

  3. codevs 1576 最长上升子序列的线段树优化

    题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...

  4. codevs 1080 线段树点修改

    先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...

  5. codevs 1082 线段树区间求和

    codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...

  6. PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树

    #44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...

  7. CF719E(线段树+矩阵快速幂)

    题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ...

  8. 【BZOJ-3779】重组病毒 LinkCutTree + 线段树 + DFS序

    3779: 重组病毒 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 224  Solved: 95[Submit][Status][Discuss] ...

  9. 【BZOJ-3673&3674】可持久化并查集 可持久化线段树 + 并查集

    3673: 可持久化并查集 by zky Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 1878  Solved: 846[Submit][Status ...

随机推荐

  1. QQ窗体的控制,同步异步打开360网盘,控制360网盘窗体的移动

     1.通过system启动飞秋进程的方式: 2.Windows下杀死进程的方式是:taskkill /f/im QQ.exe.截图例如以下: watermark/2/text/aHR0cDovL2 ...

  2. JVM虚拟机(五):JDK8内存模型—消失的PermGen

    一.JVM 内存模型 根据 JVM 规范,JVM 内存共分为虚拟机栈.堆.方法区.程序计数器.本地方法栈五个部分. 1.虚拟机栈: 每个线程有一个私有的栈,随着线程的创建而创建.栈里面存着的是一种叫“ ...

  3. OC06 -- 字典

    一. 创建不可变字典的方式: //字典的字面量,前key后value NSDictionary *dic =@{@"1":@"2",@"3" ...

  4. 自己定义TextView 调用ttf格式字体

    方法一:自己定义TextView 调用ttf格式字体 <strong>将ttf格式文件存放在assets/fonts/下</strong> 注:PC系统字体存放在C:\Wind ...

  5. 每日英语:Who Ruined The Humanities?

    You've probably heard the baleful reports. The number of college students majoring in the humanities ...

  6. 把一张图片 转成二进制流 用AFNetworking POST 上传到服务器.

    把一张图片 转成二进制流 用AFNetworking POST 上传到服务器. AFHTTPRequestOperationManager *manager = [[AFHTTPRequestOper ...

  7. layui的点击table行选中复选框

    $(document).on("click",".layui-table-body table.layui-table tbody tr",function() ...

  8. Ubuntu安装Sublime Text 2

    参考资料:http://www.technoreply.com/how-to-install-sublime-text-2-on-ubuntu-12-04-unity/ 1.去Sublime Text ...

  9. hdu 1006 Tick and Tick 有技巧的暴力

    Tick and Tick Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  10. 一款基于jQuery和CSS3炫酷3D旋转画廊特效插件

    这是一款效果炫酷的jQuery和CSS3 3D旋转画廊特效插件.该3D画廊插件可以通过前后导航按钮来切换图片,效果就像旋转木马一样.它还带有点击放大图片,显示图片标题和用键盘操作等功能. 在线预览   ...