POI 2001 Goldmine 线段树 扫描线

题目链接

http://www.acm.cs.ecnu.edu.cn/problem.php?problemid=1350

http://main.edu.pl/en/archive/oi/8/kop

求平面n个点(n<=15000),用一个 长宽为 s w的矩阵去套,能套到的最多的点数,在边上的点也算

其实跟之前矩形嵌套求面积类似 (POJ的atlantic)用类似扫描线的做法,把点当做边 (y 到  y值+w),插入线段树,这样就维护了y方向,x方向就用类似队列维护,

在距离大于s的时候,就弹出前面的(即线段树移除那条边),一边添加当前边

维护一个值,看从前扫到后,边层数累积的最多的时候即可。

一开始还以为要维护覆盖值,后来发现没用,直接一个d维护积累层数即可。

一开始输入那里没写好EOF,RE了几次,不知道什么原因,改完之后1A

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
const int N = 100020;
//int cover[N<<2];
int flag[N<<2];
int d[N<<2];
int s,w,n,maxn;
struct node{
    int x,y;
    bool operator < (const node& rhs ) const{
        if (x==rhs.x) return y<rhs.y;
        return x<rhs.x;
    }
}point[15010];
bool input()
{
    maxn=0;
    if (scanf("%d%d",&s,&w)==EOF) return false;
    scanf("%d",&n);
    for (int i=1;i<=n;i++){
        scanf("%d%d",&point[i].x,&point[i].y);
        point[i].x+=30000;
        point[i].y+=30000;
        maxn=max(maxn,point[i].y);
    }
    maxn+=w+10;
    return true;
}
void build(int rt,int l,int r)
{
    //cover[rt]=0;
    flag[rt]=0;
    d[rt]=0;
    if (l>=r) return;
    int mid=(l+r)>>1;
    build(lson);
    build(rson);
}
void pushdown(int rt,int l,int r)
{
    if (flag[rt]==0) return;
    int mid=(l+r)>>1;
    //cover[rt<<1]+=flag[rt]*(mid-l+1);
    //cover[rt<<1|1]+=flag[rt]*(r-mid);
    d[rt<<1]+=flag[rt];
    d[rt<<1|1]+=flag[rt];
    flag[rt<<1]+=flag[rt];
    flag[rt<<1|1]+=flag[rt];
    flag[rt]=0;
}
void up(int rt)
{
    //cover[rt]=cover[rt<<1]+cover[rt<<1|1];
    d[rt]=max(d[rt<<1],d[rt<<1|1]);
}
void remove(int L,int R,int rt,int l,int r)
{
    if (L<=l && r<=R){
        //cover[rt]-=(r-l+1);
        d[rt]--;
        flag[rt]+=-1;
        return;
    }
    pushdown(rt,l,r);
    int mid=(l+r)>>1;
    if (L<=mid) remove(L,R,lson);
    if (R>mid) remove(L,R,rson);
    up(rt);
}
void inserts(int L,int R,int rt,int l,int r)
{
    if (L<=l && r<=R){
        //cover[rt]+=(r-l+1);
        d[rt]++;
        flag[rt]+=1;
        return ;
    }
    pushdown(rt,l,r);
    int mid=(l+r)>>1;
    if (L<=mid) inserts(L,R,lson);
    if (R>mid) inserts(L,R,rson);
    up(rt);
}
int main()
{
    while (input()){
    sort(point+1,point+1+n);
    build(1,0,maxn);
    int pre=1;
    int ans=0;
    for (int i=1;i<=n;i++){
        //cout<<i<<endl;
        while (point[pre].x+s<point[i].x){
            remove(point[pre].y,point[pre].y+w,1,0,maxn);
            pre++;
        }
        inserts(point[i].y,point[i].y+w,1,0,maxn);
        ans=max(ans,d[1]);
      }
    printf("%d\n",ans);
   }
}

  

posted @ 2014-10-08 23:46  KRisen  阅读(337)  评论(0编辑  收藏  举报