关闭

【poj】2481 Cows【线段树单点更新】

73人阅读 评论(0) 收藏 举报
分类:

题意:

有N头牛 每头牛有一个[S,E]值 要求找出所有牛j使得Si <=Sjand Ej <= Ei and Ei - Si > Ej - Sj 对每头牛输出这样的牛的个数

题解:

先对所有牛按左端点为第一关键字升序右端点第二关键字降序排序,线段树内存右端点的值,然后每次query的是query当前节点的右端点到N的线段树和,这题还有一个就是去重,如果后一个和前一个一摸一样就直接把前面一个答案copy过来就好

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
const int N=1e5+5;
struct COW
{
    int st,ed,id;
    COW(){}
    COW(int a,int b,int c){st=a;ed=b;id=c;}
    bool operator<(const COW &b)const
    {return st<b.st||(st==b.st&&ed>b.ed);}
}cow[N];
int res[N],n,tree[N<<2],a,b;
void update(int pos,int l,int r,int rt)
{
    tree[rt]++;
    if(l==r)return;
    int mid=(l+r)>>1;
    if(pos<=mid)update(pos,lson);
    else update(pos,rson);
}
int query(int L,int R,int l,int r,int rt)
{
    if(L<=l&&r<=R)return tree[rt];
    int mid=(l+r)>>1,sum=0;
    if(L<=mid) sum+=query(L,R,lson);
    if(R>=mid) sum+=query(L,R,rson);
    return sum;
}
int main()
{
    while(scanf("%d",&n)&&n){
        for(int i=0;i<n;i++){
            scanf("%d%d",&a,&b);
            cow[i]=COW(a,b,i);
        }
        sort(cow,cow+n);
        memset(tree,0,sizeof(tree));
        for(int i=0;i<n;i++){
            if(i&&cow[i].st==cow[i-1].st&&cow[i].ed==cow[i-1].ed)
                res[cow[i].id]=res[cow[i-1].id];
            else res[cow[i].id]=query(cow[i].ed,N,0,N,1);
            update(cow[i].ed,0,N,1);
        }
        for(int i=0;i<n;i++)printf("%d%c",res[i],i==n-1?'\n':' ');
    }
    return 0;
}



0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:27158次
    • 积分:1494
    • 等级:
    • 排名:千里之外
    • 原创:132篇
    • 转载:4篇
    • 译文:0篇
    • 评论:3条