题意:
有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;
}