码迷,mamicode.com
首页 > 其他好文 > 详细

hdu6464 线段树

时间:2019-05-06 01:39:26      阅读:148      评论:0      收藏:0      [点我收藏+]

标签:long   ref   代码   turn   sort   acm   arc   区间   void   

http://acm.hdu.edu.cn/showproblem.php?pid=6464

题意

一个空序列,q次操作,一种是往序列后插入x个y,另一种是查询序列中第x小到第y小的数字之和

题解

  • 线段树维护区间和,区间个数

代码

#include<bits/stdc++.h>
#define ll long long 
using namespace std;
const int MAXN=1e5+5;
const int P=1e9+7;
ll V[MAXN<<2],W[MAXN<<2],b[MAXN],lt[MAXN],x[MAXN],y[MAXN];
int sz=0,Q[MAXN];
int tot=0;

void push_up(int o){
    V[o]=V[o<<1]+V[o<<1|1];V[o]%=P;
    W[o]=W[o<<1]+W[o<<1|1];
}
void ud(int o,int l,int r,int p,ll v){
    if(l==r){W[o]+=v;V[o]+=b[l]%P*v%P;V[o]%=P;return;}
    int mid=(l+r)/2;
    if(p<=mid)ud(o<<1,l,mid,p,v);
    else ud(o<<1|1,mid+1,r,p,v);
    push_up(o);
}
int search(int o,int l,int r,ll k){
    if(l==r){lt[++tot]=k;return l;}
    int mid=(l+r)/2;
    if(k<=W[o<<1])return search(o<<1,l,mid,k);
    else return search(o<<1|1,mid+1,r,k-W[o<<1]);
}
ll qy(int o,int l,int r,int L,int R){
    if(L>R)return 0;
    if(L<=l&&r<=R)return V[o];
    int mid=(l+r)/2;
    ll ans=0;
    if(L<=mid){ans+=qy(o<<1,l,mid,L,R);ans%=P;}
    if(R>mid){ans+=qy(o<<1|1,mid+1,r,L,R);ans%=P;}
    return ans;
}
ll qW(int o,int l,int r,int p){
    if(l==r)return W[o];
    int mid=(l+r)/2;
    if(p<=mid)return qW(o<<1,l,mid,p);
    else return qW(o<<1|1,mid+1,r,p);
}

int q;
int id(ll x){
    return lower_bound(b+1,b+sz+1,x)-b;
}
int main(){
    cin>>q;
    for(int i=0;i<q;i++){
        scanf("%d%lld%lld",&Q[i],&x[i],&y[i]);
        if(Q[i]==1)b[++sz]=y[i];
    }
    sort(b+1,b+sz+1);sz=unique(b+1,b+sz+1)-(b+1);
    //for(int i=1;i<=sz;i++)cout<<b[i]<<" ";
    //cout<<endl;
    for(int i=0;i<q;i++){
        if(Q[i]==1){
            ud(1,1,sz,id(y[i]),x[i]);
        }else{
            tot=0;
            int A=search(1,1,sz,x[i]);
            int B=search(1,1,sz,y[i]);    
            //cout<<A<<" "<<B<<endl;
            ll ans=0;
            if(A==B){ans=(lt[2]-lt[1]+1)*b[A]%P;}
            else{
                ans=qy(1,1,sz,A+1,B-1);
                ans+=(qW(1,1,sz,A)-lt[1]+1)*b[A]%P;ans%=P;
                ans+=lt[2]*b[B]%P;ans%=P;
            }
            printf("%lld\n",ans);
        }
    }
}

hdu6464 线段树

标签:long   ref   代码   turn   sort   acm   arc   区间   void   

原文地址:https://www.cnblogs.com/VIrtu0s0/p/10817219.html

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