其他分享
首页 > 其他分享> > 线段树求后继+环——cf1237D

线段树求后继+环——cf1237D

作者:互联网

/*
首先开三倍消环(两倍是不够的),倒序求值,线段树找一下后继即可 
*/
#include<bits/stdc++.h>
using namespace std;
#define N 300005

int n,a[N],ans[N];

#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
int Max[N<<2],Min[N<<2],pos1,pos2;
void build(int l,int r,int rt){
    Max[rt]=0;Min[rt]=1e9+7;
    if(l==r){
        Max[rt]=Min[rt]=a[l];
        return;
    }
    int m=l+r>>1;
    build(lson);build(rson);
    Max[rt]=max(Max[rt<<1],Max[rt<<1|1]);
    Min[rt]=min(Min[rt<<1],Min[rt<<1|1]);
}
void query1(int L,int R,int v,int l,int r,int rt){//找第一个比v大的后继
    if(l==r){
        if(Max[rt]>v)pos1=min(pos1,l);
        return;
    } 
    int m=l+r>>1;
    if(L<=l && R>=r){//可以在区间里二分了 
        if(Max[rt<<1]>v)
            query1(L,R,v,lson);
        else if(Max[rt<<1|1]>v)
            query1(L,R,v,rson);
        return;
    }
    if(L<=m)query1(L,R,v,lson);
    if(R>m)query1(L,R,v,rson); 
}
void query2(int L,int R,double v,int l,int r,int rt){//找第一个比v小的后继 
    if(l==r){
        if(Min[rt]<v)pos2=min(pos2,l);
        return;
    }
    int m=l+r>>1;
    if(L<=l && R>=r){
        if(Min[rt<<1]<v)
            query2(L,R,v,lson);
        else if(Min[rt<<1|1]<v)
            query2(L,R,v,rson);
        return;
    }
    if(L<=m)query2(L,R,v,lson);
    if(R>m)query2(L,R,v,rson);
} 

int main(){
    cin>>n;
    int mx=0,mi=0x3f3f3f3f;
    for(int i=1;i<=n;i++){
        cin>>a[i];
        a[i+n]=a[i+2*n]=a[i];
        mx=max(mx,a[i]);
        mi=min(mi,a[i]);
    }
    
    if(mx<=mi*2){
        for(int i=1;i<=n;i++)cout<<-1<<" ";
        return 0;
    }
    
    build(1,3*n,1);
    ans[3*n]=1;
    for(int i=3*n-1;i>=1;i--){
        pos1=pos2=3*n+1;
        query1(i+1,2*n,a[i],1,3*n,1);//找第一个大于a[i]的位置
        query2(i+1,3*n,1.0*a[i]/2,1,3*n,1);//找第一个小于a[i]/2的位置 
        if(pos1==3*n+1 && pos2==3*n+1)//后面都可行 
            ans[i]=3*n-i+1; 
        else if(pos1==3*n+1)//后面没有比a[i]大的 
            ans[i]=pos2-i; 
        else if(pos2==3*n+1)//后面没有比a[i]/2小的 
            ans[i]=ans[pos1]+pos1-i;
        else if(pos1<pos2)//大的在小的前面
            ans[i]=ans[pos1]+pos1-i;
        else if(pos1>pos2)
            ans[i]=pos2-i; 
    }
    for(int i=1;i<=n;i++)
        cout<<ans[i]<<" ";
} 

 

标签:rt,int,线段,树求,ans,cf1237D,pos2,rson,pos1
来源: https://www.cnblogs.com/zsben991126/p/11727773.html