【线段树求最靠前】【HDU2795】【Billboard】

题意:

有一个H*W的广告牌,当插入一个广告时(1*Wi),问最靠前的插入方式是什么


新生赛有个类似的题目,可惜当时居然没水过去。

果断用线段树做 以H为线段 建树,存[l,r]中最大的宽度,因为区间最大值满足区间和性质。


所以线段树几个要素如下:

线段:H

区间和性质:最大值


代码:

#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <ctime>
#include <algorithm>
#include <iostream>
#include <sstream>
#include <string>
#define oo 0x13131313
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define maxn 222222
using namespace std;
int h,w,n;
int tree[maxn*4];
int A[maxn];
void PushUp(int rt)
{
    tree[rt]=max(tree[rt<<1],tree[rt<<1|1]);
}
int build(int l,int r,int rt)
{
    if(l==r) {tree[rt]=w;return 0;}
    int m=(l+r)>>1;
    build(lson);
    build(rson);
    PushUp(rt);
}

void input()
{
    for(int i=1;i<=n;i++)
    scanf("%d",&A[i]);
}
int updata(int p,int k,int l,int r,int rt)
{
    int m;
    if(l==r) {tree[rt]+=k;return 0;}
    m=(l+r)>>1;
    if(p<=m) updata(p,k,lson);
    else updata(p,k,rson);
    PushUp(rt);
}
int query(int p,int l,int r,int rt)
{
    if(p>tree[rt]) return -1;
    if(l==r) return l;
    int m=(l+r)>>1;
    if(p<=tree[rt<<1]) return query(p,lson);
    else return query(p,rson);
}
void solve()
{
    for(int i=1;i<=n;i++)
    {
        int t=query(A[i],1,h,1);
        printf("%d\n",t);
        if(t!=-1)
        {
            updata(t,-A[i],1,h,1);
        }
    }
}
void init()
{
    freopen("a.in","r",stdin);
    freopen("a.out","w",stdout);
}
int main()
{
   // init();
	while(scanf("%d%d%d",&h,&w,&n)!=EOF)
    {
        memset(tree,0,sizeof(tree));
        h=min(h,200000);
        build(1,h,1);
        input();
        solve();
    }
    return 0;
}


posted on 2015-04-08 11:16  DDUPzy  阅读(130)  评论(0编辑  收藏  举报

导航