北大OJ 题目地址

其实这道题 之前也做过一次了
http://t.csdn.cn/0YZgC

不过上次是用ST 做的。这次换做 分块来实现。
【题意】
每天挤奶时,约翰的N 头奶牛(1≤N≤50,000)都以相同的顺序排队。他挑选一系列连续的奶牛来玩游戏。
为了让所有奶牛都玩得开心,它们的高度差异不应太大。约翰列出了Q组(1≤Q ≤200,000)奶牛和它们的高度(1≤height≤1,000,000)。他希望确定每个小组中最高和最矮的奶牛之间的高度差异。
【输入输出】
输入:
第1行包含两个整数N 和Q 。接下来N 行,每行都包含一个整数,表示奶牛的高度。最后Q 行,每行都包含两个整数A 和B (1≤A≤B ≤N ),代表从A 到B 的奶牛范围。
输出:
输出Q 行,每行都包含一个整数,表示该范围内最高和最矮奶牛的高度差。
【样例】

【思路分析】
这道题是典型的区间最值查询问题,可采用线段树、ST或分块解决。
【算法设计】
① 分块。划分块,记录每个元素所属的块,以及每一块的左右端点下标、最大值和最小值。
② 查询。查询[l , r ]区间最大值和最小值的差值。
【算法实现】
#include
#include
#include//max,min
#include//sqrtusing namespace std;const int inf=0x3f3f3f3f;
const int maxn=50010;
int L[maxn],R[maxn],belong[maxn],block_max[maxn],block_min[maxn];
int a[maxn],n,m;void build(){int t=sqrt(n*1.0);int num=n/t;if(n%num) num++;for(int i=1;i<=num;i++)L[i]=(i-1)*t+1,R[i]=i*t;R[num]=n;for(int i=1;i<=n;i++)belong[i]=(i-1)/t+1;for(int i=1;i<=num;i++){//求每块最值 int MIN=inf,MAX=-inf;for(int j=L[i];j<=R[i];j++){MAX=max(MAX,a[j]);MIN=min(MIN,a[j]); }block_max[i]=MAX;block_min[i]=MIN; }
}int query(int l,int r){int MIN=inf,MAX=-inf;if(belong[l]==belong[r]){for(int i=l;i<=r;i++){MAX=max(MAX,a[i]);MIN=min(MIN,a[i]); }return MAX-MIN;}else{for(int i=l;i<=R[belong[l]];i++){//左端MAX=max(MAX,a[i]);MIN=min(MIN,a[i]); }for(int i=belong[l]+1;i//中间MAX=max(MAX,block_max[i]);MIN=min(MIN,block_min[i]); }for(int i=L[belong[r]];i<=r;i++){//右端MAX=max(MAX,a[i]);MIN=min(MIN,a[i]); }}return MAX-MIN;
}int main(){int l,r;while(~scanf("%d%d",&n,&m)){for(int i=1;i<=n;i++)//下标从1开始 scanf("%d",&a[i]);build();for(int j=1;j<=m;j++){scanf("%d%d",&l,&r);printf("%d\n",query(l,r));}}return 0;
}

上一篇:数理统计笔记10:回归分析
下一篇:第十五章 图的BFS与拓扑序列