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