zoukankan      html  css  js  c++  java
  • [JSOI2018]列队(主席树)

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

    在主席树上统计答案,因为值域只有 (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
    ",query(T[l-1],T[r],k,k+r-l,1,lim));
    	}
    	return 0;
    }
    
  • 相关阅读:
    ci高级使用方法篇之连接多个数据库
    JavaSE入门学习17:Java面向对象之package(包)
    找出字符串中第一个出现次数最多的字符
    red5源代码编译并打包公布
    J.U.C--locks--AQS分析
    Spring Web MVC 原理学习(下)
    深入理解 Linux 内存管理
    Java面试问题总结
    盗版者的失落
    eclipse配置lombok插件
  • 原文地址:https://www.cnblogs.com/owencodeisking/p/10228919.html
Copyright © 2011-2022 走看看