题目:http://acm.hdu.edu.cn/showproblem.php?pid=5289
题意:给定长度为n的序列a和一个整数K,找出最大值和最小值的差值小于K的区间。输出满足条件的区间的个数。
分析:枚举a[i],以a[i]为起点,然后二分找终点(大区间满足条件的话小区间肯定也满足),依据起点和终点的位置能够算出以a[i]为起点可满足条件的区间的个数。怎么推断区间是否满足条件?能够用st算法用O(N*logN)方法进行预处理,然后O(1)查询区间最大值可最小值。先前用线段树查询超时了。。
。
后来还看到别人用树状数组+二分也过了。
。
还有的用队列写,或者直接线段树不二分。。。。
代码:
#include <iostream> #include <cstdio> using namespace std; #define lson l,m,rt<<1 #define rson m+1,r,rt<<1|1 const int maxn = 2e5; int _max,_min,n,K; int MAX[100006][20],MIN[100006][20],a[100006]; void Init() { int i,j; for(i=0;i<n;i++) MAX[i][0]=MIN[i][0]=a[i]; for(j=1;(1<<j)<=n;j++) { for(i=0;i+(1<<j)-1<n;i++) { MAX[i][j]=MAX[i][j-1]>MAX[i+(1<<(j-1))][j-1]?MAX[i][j-1]:MAX[i+(1<<(j-1))][j-1]; MIN[i][j]=MIN[i][j-1]<MIN[i+(1<<(j-1))][j-1]?MIN[i][j-1]:MIN[i+(1<<(j-1))][j-1]; } } } bool ok(int L,int R) { int k=0; while((1<<(k+1))<=(R-L+1)) k++; _min=MIN[L][k]<MIN[R-(1<<k)+1][k]?MIN[L][k]:MIN[R-(1<<k)+1][k]; _max=MAX[L][k]>MAX[R-(1<<k)+1][k]?MAX[L][k]:MAX[R-(1<<k)+1][k]; return _max-_min<K; } int Find(int s) { int down=s+1,up=n-1,mid,ret=s; while(down<=up) { mid=(down+up)>>1; if(!ok(s,mid)) up=mid-1; else { down=mid+1; if(ret<mid) ret=mid; } } return ret; } int main() { int ncase,i,j; long long ans; scanf("%d",&ncase); while(ncase--) { scanf("%d%d",&n,&K); for(i=0;i<n;i++) scanf("%d",&a[i]); Init(); ans=n; for(i=0;i<n;i++) ans+=Find(i)-i; printf("%I64d ",ans); } return 0; }