HDU 5316 线段树区间最值问题

T组数据

N个数字,M次操作

op=0:找出L-R区间内的最大‘值’

op=1:把a位置的数字换成b

对最大‘值’的定义:取区间内的最大子序列,需要保证子序列的下标为奇偶交替的


用线段树分别记录每个区间的

ee:以偶数下标开始偶数下标结束的最大和

eo:以偶数下标开始奇数下标结束的最大和

oe:以奇数下标开始偶数下标结束的最大和

oo:以奇数下标开始奇数下标结束的最大和


对每次询问分别处理4种情况的最大值即可

修改时单点更新即可


#include "stdio.h"
#include "string.h"
const __int64 inf=0x3f3f3f3f3f3f3f3f;
__int64 ans,a[100100];

struct node
{
    int l,r;
    __int64 ee,eo,oe,oo;
}data[400010];

__int64 Max(__int64 a,__int64 b)
{
    if (a<b) return b;else return a;
}

void Pushup(int k)
{
    data[k].ee=Max(-inf,Max(data[k*2].ee+data[k*2+1].oe,data[k*2].eo+data[k*2+1].ee));
    data[k].ee=Max(data[k].ee,data[k*2].ee);
    data[k].ee=Max(data[k].ee,data[k*2+1].ee);

    data[k].oo=Max(-inf,Max(data[k*2].oo+data[k*2+1].eo,data[k*2].oe+data[k*2+1].oo));
    data[k].oo=Max(data[k].oo,data[k*2].oo);
    data[k].oo=Max(data[k].oo,data[k*2+1].oo);

    data[k].eo=Max(-inf,Max(data[k*2].ee+data[k*2+1].oo,data[k*2].eo+data[k*2+1].eo));
    data[k].eo=Max(data[k].eo,data[k*2].eo);
    data[k].eo=Max(data[k].eo,data[k*2+1].eo);

    data[k].oe=Max(-inf,Max(data[k*2].oo+data[k*2+1].ee,data[k*2].oe+data[k*2+1].oe));
    data[k].oe=Max(data[k].oe,data[k*2].oe);
    data[k].oe=Max(data[k].oe,data[k*2+1].oe);
}

void build(int l,int r,int k)
{
    int mid;
    data[k].l=l;
    data[k].r=r;
    if (l==r)
    {
        if (l%2==1)
        {
            data[k].oo=a[l];
            data[k].oe=data[k].eo=data[k].ee=-inf;
        }
        else
        {
            data[k].ee=a[l];
            data[k].oe=data[k].eo=data[k].oo=-inf;
        }
        return ;
    }

    mid=(l+r)/2;

    build(l,mid,k*2);
    build(mid+1,r,k*2+1);

    Pushup(k);
}

void updata(int n,int x,int k)
{
    if (data[k].l==n && data[k].r==n)
    {
        if (data[k].l%2==1)
        {
            data[k].oo=x;
            data[k].eo=data[k].oe=data[k].ee=-inf;
        }
        else
        {
            data[k].ee=x;
            data[k].eo=data[k].oe=data[k].oo=-inf;
        }
        return ;
    }

    if (n<=data[k*2].r) updata(n,x,k*2);
    else updata(n,x,k*2+1);

    Pushup(k);
}

node search(int l,int r,int k)
{
    node temp,t1,t2;
    int mid;
    if (data[k].l==l && data[k].r==r)
    {
        temp.ee=data[k].ee;
        temp.oo=data[k].oo;
        temp.eo=data[k].eo;
        temp.oe=data[k].oe;
        return temp;
    }

    mid=(data[k].l+data[k].r)/2;
    if (r<=mid) return search(l,r,k*2);
    else
        if (l>mid) return search(l,r,k*2+1);
    else
    {
        t1=search(l,mid,k*2);
        t2=search(mid+1,r,k*2+1);
        temp.ee=Max(Max(Max(t1.ee+t2.oe,t1.eo+t2.ee),t1.ee),t2.ee);
        temp.eo=Max(Max(Max(t1.eo+t2.eo,t1.ee+t2.oo),t1.eo),t2.eo);
        temp.oo=Max(Max(Max(t1.oo+t2.eo,t1.oe+t2.oo),t1.oo),t2.oo);
        temp.oe=Max(Max(Max(t1.oo+t2.ee,t1.oe+t2.oe),t1.oe),t2.oe);
        return temp;
    }
}

int main()
{
    int t,n,m,i,op,l,r;
    node mark;
    scanf("%d",&t);
    while (t--)
    {
        scanf("%d%d",&n,&m);
        for (i=1;i<=n;i++)
            scanf("%I64d",&a[i]);
        build(1,n,1);
        while (m--)
        {
            scanf("%d%d%d",&op,&l,&r);
            if (op==0)
            {
                ans=-inf;
                mark=search(l,r,1);
                ans=Max(ans,mark.oo);
                ans=Max(ans,mark.ee);
                ans=Max(ans,mark.eo);
                ans=Max(ans,mark.oe);

                printf("%I64d\n",ans);
            }
            else
                updata(l,r,1);
        }
    }
    return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值