题意:插队问题;
n个操作,操作为: pos,val
表示把一个人插到pos位置并且这个人特征值记为val,最后输出整个序列
有一点是,每个pos[i] 的范围规定了是 1到i ,这个限定使得题目可以比较容易的解决,显然因此,对于i来说,前i-1个人必然都排在前i-1个位置。因此影响第i个人的位置的 是看【i以后的人是否要插入pos[i]之前的位置】
即如果我们逆序来看,对于每个pos[i],val,当前这个人本来应该是留在pos[i]位置,但是如果他之前还有X人在pos[i]的前面插入了,则当前这个人最终的位置会是 pos[i]+X。
所以线段树的节点记录 所管辖区间 有多少个人,判断当前人是否要排在左区间,则判断 if (pos[i] + sum[rt<<1] 《 mid ) ;
如果进入右区间的话, 则pos需要变为pos+sum[rt<<1]
#include <cstdio>
#include <cmath>
#include <cstring>
#include <string>
#include <algorithm>
#include <queue>
#include <map>
#include <set>
#include <vector>
#include <iostream>
using namespace std;
const long long maxn = 200005;
int ans[200005];
struct tree
{
int sum[maxn*4];
void init()
{
memset(sum,0,sizeof(sum));
}
void pushup(int rt)
{
sum[rt]= (sum[rt<<1]+sum[rt<<1|1]);
}
int query(int pos,int val,int l,int r,int rt) //查询大于等于val的第一个位置
{
if (r==l)
{
ans[l]=val;
sum[rt]=1;
return l;
}
int mid=(l+r)>>1;
if (sum[rt<<1]+pos<=mid)
query(pos,val,l,mid,rt<<1);
else
query(pos+sum[rt<<1],val,mid+1,r,rt<<1|1);
pushup(rt);
}
};
tree tp;
int num[200005];
int val[200005];
int main( )
{
int i;
int x;
int n,h,w;
int cun=0;
while(scanf("%d",&n)!=EOF)
{
tp.init();
for (i=1; i<=n; i++)
{
scanf("%d %d",&num[i],&val[i]);
}
for (i=n; i>=1; i--)
{
int pos=num[i];
int v=val[i];
int ret= tp.query(pos+1,v,1,n,1);
}
for (i=1; i<=n; i++)
{
if (i!=1) printf(" ");
printf("%d",ans[i]);
}
printf("\n");
}
return 0;
}