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

题意:

有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;
}



阅读更多
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/a709743744/article/details/51550155
个人分类: ACM_线段树
上一篇【hdu】3231 Box Relations【拓扑排序三维】
下一篇【hdu】4228 Flooring Tiles【反素数】
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

关闭
关闭