参考博客:

Mahr

Liangyj

线段树:是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点。
使用线段树可以快速的查找某一个节点在若干条线段中出现的次数,时间复杂度为O(logN)。而未优化的空间复杂度为2N,实际应用时一般还要开4N的数组以免越界,因此有时需要离散化让空间压缩。

建树操作
struct node
{
    long long sum,l,r,tag;
}t[666666];
long long num[666666];
void build(int p,int l,int r)
{
    t[p].l=l;
    t[p].r=r;
    if(l==r)
    {
        t[p].sum=num[l];
        return ;
    }
    int mid=(l+r)/2;
    build(2*p,l,mid);
    build(2*p+1,mid+1,r);
    t[p].sum=t[2*p].sum+t[2*p+1].sum;
}
懒标记(区间求和)
void lazy(int p)
{
    if(t[p].tag)
    {
        t[2*p].sum+=t[p].tag*(t[p*2].r-t[p*2].l+1);
        t[2*p+1].sum+=t[p].tag*(t[p*2+1].r-t[p*2+1].l+1);
        t[2*p].tag+=t[p].tag;
        t[2*p+1].tag+=t[p].tag;
        t[p].tag=0;
    }
}
区间修改(加/减)
void change(int p,int x,int y,int z)
{
    if(t[p].l>=x&&t[p].r<=y)
    {
        t[p].sum+=1LL*z*(t[p].r-t[p].l+1);
        t[p].tag+=z;
        return ;
    }
    int mid=(t[p].l+t[p].r)/2;
    lazy(p);
    if(x<=mid)
    {
        change(p*2,x,y,z);
    }
    if(y>mid)
    {
        change(p*2+1,x,y,z);
    }
    t[p].sum=t[2*p].sum+t[p*2+1].sum;
    return ;
}
区间求和
long long sum(int p,int x,int y)
{
    if(t[p].l>=x&&t[p].r<=y)
    {
        return t[p].sum;
    }
    int mid=(t[p].l+t[p].r)/2;
    long long ans=0;
    lazy(p);
    if(x<=mid)
    {
        ans+=sum(p*2,x,y);
    }
    if(mid<y)
    {
        ans+=sum(p*2+1,x,y);
    }
    return ans;
}
单点查询
long long single_sum(int p,int x)
{
    {
        if(t[p].l==t[p].r&&t[p].l==x)
        {
            return t[p].sum;
        }
        int mid=(t[p].l+t[p].r)/2;
        lazy(p);
        long long ans=0;
        if(mid>=x)
        {
            ans+=single_sum(2*p,x);
        }
        if(mid<x)
        {
            ans+=single_sum(2*p+1,x);
        }
        return ans;
    }
}
单点修改(加/减)
void single_change (int p,int x,int z)
{
    if(t[p].l==t[p].r&&t[p].l==x)
    {
        t[p].sum+=z;
        return ;
    }
    int mid=(t[p].l+t[p].r)/2;
    if(mid>=x)
    {
        single_change(2*p,x,z);
    }
    if(mid<x)
    {
        single_change(2*p+1,x,z);
    }
    t[p].sum=t[p*2].sum+t[p*2+1].sum;
}
区间修改(更改数值)
void lazy(int p)
{
    if(t[p].tag)
    {
        t[2*p].sum=t[p].tag*(t[p*2].r-t[p*2].l+1);//少了一个+号
        t[2*p+1].sum=t[p].tag*(t[p*2+1].r-t[p*2+1].l+1);//少了一个+号
        t[2*p].tag=t[p].tag;
        t[2*p+1].tag=t[p].tag;
        t[p].tag=0;
    }
}
void change(int p,int x,int y,int z)
{
    if(t[p].l>=x&&t[p].r<=y)
    {
        t[p].sum=1LL*z*(t[p].r-t[p].l+1);//少了一个+号
        t[p].tag=z;
        return ;
    }
    int mid=(t[p].l+t[p].r)/2;
    lazy(p);
    if(x<=mid)
    {
        change(p*2,x,y,z);
    }
    if(y>mid)
    {
        change(p*2+1,x,y,z);
    }
    t[p].sum=t[2*p].sum+t[p*2+1].sum;
    return ;
}
区间最大值,最小值
const int maxn = 200000+200;
struct node
{
    int l,r,maxx,minn;
}t[maxn<<2];
int num[maxn];
void build(int p,int l,int r)
{
    t[p].l=l;
    t[p].r=r;
    if(l==r)
    {
        t[p].maxx=num[l];
        t[p].minn=num[l];
        return ;
    }
    int mid=(l+r)>>1;
    build(p<<1,l,mid);
    build(p<<1|1,mid+1,r);
    t[p].maxx=max(t[p<<1].maxx,t[p<<1|1].maxx);
    t[p].minn=min(t[p<<1].minn,t[p<<1|1].minn);
}
int search_max(int pos,int x,int y)
{
    if(x<=t[pos].l&&y>=t[pos].r)
    {
        return t[pos].maxx;
    }
    int ans=-1;
    int mid=(t[pos].l+t[pos].r)>>1;
    if(x<=mid) ans=max(ans,search_max(pos<<1,x,y));
    if(y>mid) ans=max(ans,search_max(pos<<1|1,x,y));
    return ans;
}
int search_min(int pos,int x,int y)
{
    if(x<=t[pos].l&&y>=t[pos].r)
    {
        return t[pos].minn;
    }
    int ans=10000000+1;
    int mid=(t[pos].l+t[pos].r)>>1;
    if(x<=mid) ans=min(ans,search_min(pos<<1,x,y));
    if(y>mid) ans=min(ans,search_min(pos<<1|1,x,y));
    return ans;
}
文章作者: Hao.Jia
版权声明: 本站所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 Hao.Jia's Blog
算法 算法碎片
喜欢就支持一下吧