不用二分的借教室

  在网上一直看到用二分做的借教室,说什么线段树会惨遭TLE,然后我就试了一下,并没有什么事情发生(或许是因为optimizi(2)...),但并没有什么关系!!!

  我们只需要在每个树的节点上打上mi的标记,表示其子树的最小值,在更新的时候如果mi小于0,就知道不行了,标记一下直接输出就好了。。。

  详见代码:

#include<bits/stdc++.h>
#pragma GCC optimize(2)
#define in1(x) scanf("%d",&x)
#define in3(x,y,z) scanf("%d%d%d",&x,&y,&z)
using namespace std;
int a[4000000],n,m,k,x,y,day,shu,f,val;
struct st{
    int l,r,val,add,mi;
}tr[4000000];
void build(int l,int r,int k)
{
    tr[k].l = l;
    tr[k].r = r;
    if(l==r)
    {
        tr[k].val = a[l];
        tr[k].mi = tr[k].val;
        return;
    }
    int mid = (l+r)>>1;
    build(l,mid,k<<1);
    build(mid+1,r,k<<1|1);
    tr[k].val = tr[k<<1].val + tr[k<<1|1].val;
    tr[k].mi = min(tr[k<<1].mi , tr[k<<1|1].mi);
}
void push(int k)
{
    int l = tr[k].l , r = tr[k].r;
    int w = tr[k>>1].add;
    tr[k].val += (r-l+1)*w;
    tr[k].add += w;
    tr[k].mi -= w;
}
int update(int x,int y,int k,int val)
{
    int l = tr[k].l , r = tr[k].r;
    if(tr[k].add)
    {
        push(k<<1);
        push(k<<1|1);
        tr[k].add = 0;
    }
    if(y<l || r<x) return 0;
    if(x<=l && r<=y)
    {
        tr[k].add += val;
        tr[k].val -= (r-l+1)*val;
        tr[k].mi -=val;
        if(tr[k].mi < 0) f=1;
        return 0;
    }
    int mid = (l+r)>>1;
    if(x<=mid)
        update(x,y,k<<1,val);
    if(y>mid)
        update(x,y,k<<1|1,val);
    tr[k].mi = min(tr[k<<1].mi , tr[k<<1|1].mi);
    if(tr[k].mi < 0) f=1;
    return 0;
}
main(){
    in1(k);
    in1(shu);
    for(int i=1;i<=k;i++)
        in1(a[i]);
    build(1,k,1);
    for(int i=1;i<=shu;i++)
    {
        in3(day,x,y);
        val=day;
        update(x,y,1,val);
        if(f==1) 
        {
            printf("-1\n%d",i);
            return 0;
        }
    }
    printf("0");
}

 

posted @ 2017-08-19 21:41  cc123321  阅读(204)  评论(0编辑  收藏  举报