码迷,mamicode.com
首页 > Web开发 > 详细

[JSOI2018]列队(主席树)

时间:2019-01-06 15:00:46      阅读:181      评论:0      收藏:0      [点我收藏+]

标签:\n   signed   long   i++   std   pre   不用   define   tchar   

跟上次那道列队不一样,但都是九条可怜。。。(吉老师太强了)

在主席树上统计答案,因为值域只有 \(10^6\) 甚至不用离散化。。。

\(Code\ Below:\)

#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn=500000+10;
const int lim=1000000;
const int inf=0x3f3f3f3f;
int n,m,a[maxn],Sum[maxn],ans,T[maxn],L[maxn<<5],R[maxn<<5],sum[maxn<<5],siz[maxn<<5],cnt;

inline int read(){
    register int x=0,f=1;char ch=getchar();
    while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
    while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
    return (f==1)?x:-x;
}

void update(int &now,int pre,int l,int r,int x){
    now=++cnt;L[now]=L[pre];R[now]=R[pre];
    sum[now]=sum[pre]+x;siz[now]=siz[pre]+1;
    if(l == r) return ;
    int mid=(l+r)>>1;
    if(x <= mid) update(L[now],L[pre],l,mid,x);
    else update(R[now],R[pre],mid+1,r,x);
}

int query(int u,int v,int Le,int Ri,int l,int r){
    if(Le > Ri) return 0; 
    if(r <= Ri) return (Le+Ri)*(Ri-Le+1)/2-(sum[v]-sum[u]);
    if(Le <= l) return (sum[v]-sum[u])-(Le+Ri)*(Ri-Le+1)/2;
    int mid=(l+r)>>1,cnt=siz[L[v]]-siz[L[u]];
    return query(L[u],L[v],Le,Le+cnt-1,l,mid)+query(R[u],R[v],Le+cnt,Ri,mid+1,r);
}

signed main()
{
    n=read(),m=read();
    for(int i=1;i<=n;i++){
        a[i]=read();
        update(T[i],T[i-1],1,lim,a[i]);
    }
    int l,r,k;
    while(m--){
        l=read(),r=read(),k=read();
        printf("%lld\n",query(T[l-1],T[r],k,k+r-l,1,lim));
    }
    return 0;
}

[JSOI2018]列队(主席树)

标签:\n   signed   long   i++   std   pre   不用   define   tchar   

原文地址:https://www.cnblogs.com/owencodeisking/p/10228919.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!