poj-2828-Buy Tickets-线段树-单点

题意:插队问题;


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


你可能感兴趣的:(poj-2828-Buy Tickets-线段树-单点)