hdu 5306 Gorgeous Sequence(暴力线段树)(好题)

题意:区间最大长度为1000000,

        三种操作:

          区间求和;

          区间求最大值;

          将节点值更新为当前值与给定值中的最小值(有趣的更新);

思路: 暴力线段树。关键在于处理标记,需要维护最大标记,标记覆盖范围,所在区间;

        覆盖区域标记起到很关键的作用;

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
struct node{
  long long sum;
  int tag,maxtag,mmax,cover,s;
}q[5000100];
int num[5000100];
node  pushdown(node c,int t)  //处理标记及标记覆盖范围
{
    if(!t) return c;
    if(c.cover!=c.s) c.mmax=t;
    c.maxtag=t;
    c.sum+=1LL*(c.s-c.cover)*t;
    c.cover=c.s;
    return c;
}
node pushup(node a,node b,int t) //向上更新
{
    node c;
    c.tag=t;
    c.s=a.s+b.s;
    c.sum=a.sum+b.sum;
    c.cover=a.cover+b.cover;
    c.mmax=max(a.mmax,b.mmax);
    c.maxtag=max(a.maxtag,b.maxtag);
    return pushdown(c,t);
}
void build(int l,int r,int pos)
{
    q[pos].tag=0;
    q[pos].s=r-l+1; //区间长度
    if(l<r){
        int mid=(l+r)/2;
        build(l,mid,pos*2);
        build(mid+1,r,pos*2+1);
        q[pos]=pushup(q[pos*2],q[pos*2+1],0);
    }
    else{
        q[pos].mmax=q[pos].sum=q[pos].maxtag=q[pos].tag=num[l];
        q[pos].cover=1;
    }
}
void cl(int k,int t)
{
    if(q[k].maxtag<t) return;//若已经是最小,则返回
    if(q[k].tag>=t){  
        q[k].tag=0;
    }
    if(q[k].s>1){  //区间长度大于1
        cl(k*2,t);
        cl(k*2+1,t);
    }
    if(q[k].s==1){ //处理点
        q[k].sum=q[k].mmax=q[k].maxtag=q[k].tag;
        q[k].cover=(q[k].tag>0); //决定是否覆盖
    }
    else{ //更新
        q[k]=pushup(q[k*2],q[k*2+1],q[k].tag);
    }
}
void update(int L,int R,int t,int l,int r,int pos)
{
    if(q[pos].tag&&q[pos].tag<=t) return; //已是最小,则不用update
    if(L<=l&&r<=R){  
        cl(pos,t);  //区间处理
        q[pos].tag=t;
        if(l==r){
            q[pos].mmax=q[pos].maxtag=q[pos].sum=q[pos].tag=t;
            q[pos].cover=(q[pos].tag>0); //是否覆盖
        }
        else{
            q[pos]=pushup(q[pos*2],q[pos*2+1],t);
        }
    }
    else{
        int mid=(l+r)/2;
        if(R<=mid) update(L,R,t,l,mid,pos*2);
        else if(mid<L) update(L,R,t,mid+1,r,pos*2+1);
        else update(L,R,t,l,mid,pos*2),update(L,R,t,mid+1,r,pos*2+1);
        q[pos]=pushup(q[pos*2],q[pos*2+1],q[pos].tag);
    }
}
node query(int L,int R,int l,int r,int pos)
{
    if(L<=l&&r<=R){
      return q[pos];
    }
    int mid=(l+r)/2; 
    if(R<=mid) return pushdown(query(L,R,l,mid,pos*2),q[pos].tag); //查询时,区间处理
    if(mid<L) return pushdown(query(L,R,mid+1,r,pos*2+1),q[pos].tag);
    return pushup(query(L,R,l,mid,pos*2),query(L,R,mid+1,r,pos*2+1),q[pos].tag);
}
int main()
{
    int t,n,m,i,j,k,temp;
    int a,b,c;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d",&n,&m);
        for(i=1;i<=n;i++)
            scanf("%d",&num[i]);
        build(1,n,1);
        for(i=0;i<m;i++)
        {
            scanf("%d",&temp);
            if(temp==0){
              scanf("%d%d%d",&a,&b,&c);
              update(a,b,c,1,n,1);
            }
            else if(temp==1){
              scanf("%d%d",&a,&b);
              printf("%d\n",query(a,b,1,n,1).mmax);
            }
            else{
              scanf("%d%d",&a,&b);
              printf("%lld\n",query(a,b,1,n,1).sum);
            }
        }
    }
    return 0;
}

 

你可能感兴趣的:(sequence)