HDU - 3486(RMQ o(1)查询)

首先,要说明该题不满足二分性质。

但是不用二分过不了该题啊,RMQ的o(1)查询,还有二分时上线设为n和n+1交上去结果不同,只能说明二分不满足单调性。

#include <stdio.h>
#include <cmath>
#define max(a,b) a>b?a:b
#define min(a,b) a<b?a:b
const int N=200105;
int n,Q,c[N],a,b;
int st[N][20],lg2[N]; //20不一定是唯一的。需要计算log(N)/log(2)
void ST(int *a,int n)
{
    lg2[0]=-1;
    for(int i=1;i<=n;i++)
        lg2[i]=lg2[i-1]+(i&(i-1)?0:1);
    for(int i=0;i<n;i++) st[i][0]=a[i];
    for(int j=1;j<=lg2[n];j++)
        for(int i=0;lg2[n-i]>=j;i++)
            st[i][j]=max(st[i][j-1],st[i+(1<<(j-1))][j-1]);
}
int RMQ(int x,int y)
{
    int k=lg2[y-x+1];
    return max(st[x][k],st[y-(1<<k)+1][k]);
}
int judge(int num,int len){
   int st=1;
   int sum=0;
   for(int i=1;i<=num;i++){
     sum+=RMQ((i-1)*len,i*len-1);
     if(sum>Q) return 1;
   }
   return 0;
}
int main()
{
    while(scanf("%d%d",&n,&Q)==2 && n>=0){
    int sum=0;
    for(int i=0;i<n;i++){
        scanf("%d",&c[i]);
        sum+=c[i];
    }
    if(sum<=Q) {printf("-1\n"); continue;}
    ST(c,n);
    int x=1,y=n;
    while(x<y){
         int mid=(x+y)/2;
         if(judge(mid,n/mid)) y=mid;
         else x=mid+1;
    }
    printf("%d\n",x);
    }
    return 0;
}


你可能感兴趣的:(HDU - 3486(RMQ o(1)查询))