首页 > 其他分享 >线段树

线段树

时间:2023-03-04 14:44:19浏览次数:29  
标签:int tr 线段 modify mid build ll

使用线段树进行多点更改和多点查询

建树

#include<iostream>

using namespace std;
typedef long long ll;

const int N=2e5+10;

struct segm
{
    int l,r;
    ll v;
} tr[N*4];
ll a[N];

void push_up(int u)
{
    tr[u].v=tr[u>>1].v+tr[u>>1|1].v;
}

//建树
void build(int u,int l,int r)
{
    //叶子节点
    if(l==r)
    {
        tr[u]={l,r,a[l]};
        return ;
    }
    tr[u]={l,r};
    int mid=l+r>>1;
    build(u>>1,l,mid),build(u>>1|1,mid+1,r);
    push_up(u);
}

//单点修改
void modify(int u,int x,ll d)
{
    if(tr[u].l==tr[u].r)
    {
        tr[u].v=d;
        return ;
    }

    int mid=tr[u].l+tr[u].r>>1;
    if(mid>=x) modify(u>>1,x,d);
    else modify(u>>1|1,x,d);
    push_up(u);
}

//多点修改
void multi_modify(int u,int l,int r,ll d)
{
    if(tr[u].l==tr[u].r)
    {
        tr[u].v=d;
        return;
    }

    int mid=tr[u].l+tr[u].r>1;
    if(mid>=l) multi_modify(u>>1,l,r,d);
    if(mid<r) multi_modify(u>>1|1,l,r,d);
    push_up(u);
}

//区域查询
ll query(int u,int l,int r)
{
    if(tr[u].l>=l&&tr[u].r<=r)
    {
        return tr[u].v;
    }
    ll sum=0;
    int mid=tr[u].l+tr[u].r>>1;
    if(mid>=l) sum=query(u>>1,l,r);
    if(mid>r) sum+=query(u>>1|1,l,r);
    return sum;
}

int main()
{
    build(1,1,n);
}

标签:int,tr,线段,modify,mid,build,ll
From: https://www.cnblogs.com/ccag/p/17178268.html

相关文章