题目:http://poj.org/problem?id=2828
大意:一群人排队,第i个人来到队伍中站到处于posi的人的右边,且每个人都有不同的表示值,问最终的结果?
Sample Input
4 0 77 1 51 1 33 2 69 4 0 20523 1 19243 1 3890 0 31492
Sample Output
77 33 69 51 31492 20523 3890 19243
Hint
The figure below shows how the Little Cat found out the final order of people in the queue described in the first test case of the sample input.
#include <iostream> #include <cstdio> using namespace std; const int maxn=2e5+5; int ans[maxn]; struct node{ int l,r,num; int mid(){ return (l+r)/2; } }tree[maxn<<2]; void build(int root,int low,int high){ tree[root].l=low; tree[root].r=high; if(low==high){ tree[root].num=1; return ; } int m=tree[root].mid(); build(2*root,low,m); //no if else build(2*root+1,m+1,high); tree[root].num=tree[2*root].num+tree[2*root+1].num; } void insert(int root,int p,int val){ if(tree[root].l==tree[root].r){ //cout<<tree[root].l<<endl; tree[root].num=0; ans[tree[root].l]=val; return ; } if(p<=tree[2*root].num) insert(2*root,p,val); else { p=p-tree[2*root].num; //pos有一个作用:测试num,pos-num变成右子树的测试num insert(2*root+1,p,val); //eg.p=3,left_num=1,for right_tree: p=2 } //if p=3,left_tree is full,left_num=0, find suitable place in right_tree. tree[root].num=tree[2*root].num+tree[2*root+1].num; } struct tnode{ int pos,val; }tp[maxn]; int main() { //freopen("cin.txt","r",stdin); int n; while(cin>>n){ build(1,1,n); for(int i=0;i<n;i++){ scanf("%d%d",&tp[i].pos,&tp[i].val); } //for(int i=7;i>=1;i--)cout<<tree[i].num<<" "; cout<<endl; for(int i=n-1;i>=0;i--){ insert(1,tp[i].pos+1,tp[i].val); } //for(int i=7;i>=1;i--)cout<<tree[i].num<<" "; cout<<endl; for(int i=1;i<n;i++){ printf("%d ",ans[i]); } printf("%d\n",ans[n]); } return 0; }