题意:
给出n个点的座标值,计算出每个点的位于该点左下方的点的个数level(包括x或y相等的边界),最后统计level从0到n-1的点的个数并输出
算法:
首先将n个点排序,按照y从小到大,y相等时按x从小到大排序
建立树状数组c[],该树状数组的求和功能getsum(i)是求当前已经插入的点的座标的x值小于等于i的点的个数,即为该点的level值+1
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
using namespace std;
class node
{
public:
int x,y;
int level;
}p[15005];//输入的点
int c[32006];//树状数组
int n;//输入的点的个数
bool cmp(node a,node b)
{
if(a.y==b.y)
{
return a.x<=b.x;
}
return a.y<b.y;
}
int lowbit(int x)
{
return x&(-x);
}
void insert(int x)
{
while(x<=32004)
{
c[x]++;
x+=lowbit(x);
}
}
int getsum(int x)
{
int ans=0;
while(x>=1)
{
ans+=c[x];
x-=lowbit(x);
}
return ans;
}
int result[15005];//最后结果
int main()
{
cin>>n;
for(int i=1;i<=n;i++)
{
cin>>p[i].x>>p[i].y;
p[i].x++;//因为输入数据中有0,而树状数组的处理过程中是不能有0的,所以进行+1
p[i].y++;
}
memset(c,0,sizeof(c));
sort(p+1,p+n+1,cmp);//排序
for(int i=1;i<=n;i++)
{
insert(p[i].x);
p[i].level=getsum(p[i].x)-1;
}
memset(result,0,sizeof(result));
for(int i=1;i<=n;i++)
{
result[p[i].level]++;
}
for(int i=0;i<n;i++)
{
cout<<result[i]<<endl;
}
//system("PAUSE");
return 0;
}