zoukankan      html  css  js  c++  java
  • POJ 2828

      题目传送门 http://poj.org/problem?id=2828

    题目大意:每次插队的时候插到第$POs[i]$个人后面去,最后打印出最终的排列,(好暴力的插队方式啊~~

    在中国都很少见呢~ ><

    题目思路:从后面往前遍历,一个一个将那个人插到还有的第$POS[i] + 1$个空格那里,神奇的思路,在线段树上每个节点有如下变量

    $numquad$用于维护该区间还剩下的空格数

    $valuequad$用于记录该点的value值,和题意相同,不过这个变量只用于线段树的叶子

    $idquad$用于维护原序列

    第一次写线段树只看了思路然后就AC的,完全没参考别人的代码…….看来我对线段树的了解上了一个层次了233, $$Butquad Still quad Longquad Longquad Waitquad Toquad Go!!$$%

    #include <cstdio>
    #include <algorithm>
    
    using namespace std;
    #define lson l,m,rt<<1
    #define rson m+1,r,rt<<1|1
    #define ll rt<<1
    #define rr rt<<1|1
    const int maxn = 222222;
    struct node{
        int id,value,num;
    };
    int n;
    node get[maxn];
    node sum[maxn<<2];
    void PushUp(int rt) {
        sum[rt].num = sum[ll].num + sum[rr].num;
    }
    void build(int l,int r,int rt) {
        if(l == r) {
            sum[rt].num = 1;
            sum[rt].id = 0;
            sum[rt].value = 0;
            return ;
        }
        int m = (l + r)>>1;
        build(lson);
        build(rson);
        PushUp(rt);
    }
    void update(int id,int num,int value,int l,int r,int rt) {
        if(l == r) {
            sum[rt].value = value;
            sum[rt].num   = 0;
            sum[rt].id    = id;
            return ;
        }
        int m = (l + r)>>1;
        if(num <= sum[ll].num) update(id,num,value,lson);
        else update(id,num - sum[ll].num,value,rson);
        PushUp(rt);
    }
    int ans_num;
    void print(int l,int r,int rt) { //i = 1;
        if(l == r&&ans_num == n) {
            printf("%d
    ",get[sum[rt].id].value);
            return ;
        }else if(l == r) {
            printf("%d ",get[sum[rt].id].value);
            ans_num++;
            return ;
        }
        int m = (l + r) >> 1;
        print(lson);
        print(rson);
    }
    int main() {
        while(~scanf("%d",&n)) {
            for(int i = 1;i <= n;++i) {
                scanf("%d%d",&get[i].num,&get[i].value);
                get[i].id = i;
            }
            build(1,n,1);
            //printf("%d
    ",sum[1].num);
            for(int i = n;i >= 1;--i) {
                update(i,get[i].num+1,get[i].value,1,n,1);
            }
            ans_num = 1;
            print(1,n,1);
        }
        return 0;
    }
  • 相关阅读:
    简单的进度条程序
    python装饰器
    冒泡,递归
    实现随机验证码
    三木运算,三元运算
    set()集合
    list()列表
    python3.5 Str类型与bytes类型转换
    说说编码与转义的区别
    Java 中字符编码相关的问题解析
  • 原文地址:https://www.cnblogs.com/jusonalien/p/4128119.html
Copyright © 2011-2022 走看看