题目大意

一些小朋友在排队,每次来一个人,第i个人会插到第x个人的后面。权值为y。保证x∈[0,i-1]。

按照最后的队伍顺序,依次输出每个人的权值。

解题分析

好气吖。本来是在做splay练习,然后发现这道题用splay写T掉了,可能是我的splay常数太大了吧。要不要考虑去学一下自顶向下建树的splay,据说会快一点。

可以倒着考虑问题。如果倒着安排小朋友的队伍的话,就不用考虑插队的问题了。如果第i个人插到了第x个人的后面。

用线段树写的话,记录一下空格的数量,每次找个第x+1个空格的位置插入值。

用树状数组写的话,记录一下空格的数量,每次二分找出第x+1空格的位置插入值。

参考程序

Splay T掉了= =

#include <cstdio>
using namespace std; class Splay_tree
{
private:
struct node
{
int val,sz;
node *l,*r,*f;
node(int _val=-1,int _sz=1,node*_f=NULL,node*_l=NULL,node*_r=NULL):
val(_val),sz(_sz),f(_f),l(_l),r(_r){}
};
node *rt;
void del(node *x)
{
if (!x) return;
del(x->l); del(x->r);
delete x;
}
void pushup(node *x)
{
x->sz=1;
if (x->l) x->sz += x->l->sz;
if (x->r) x->sz += x->r->sz;
}
void left(node *x,node *&rt)
{
node *y=x->f,*z=y->f;
if (y==rt) rt=x; else if (y==z->l) z->l=x; else z->r=x;
if (x->l) x->l->f=y; y->f=x; x->f=z;
y->r=x->l; x->l=y;
pushup(y); pushup(x);
}
void right(node *x,node *&rt)
{
node *y=x->f,*z=y->f;
if (y==rt) rt=x; else if (y==z->l) z->l=x; else z->r=x;
if (x->r) x->r->f=y; y->f=x; x->f=z;
y->l=x->r; x->r=y;
pushup(y); pushup(x);
}
void splay(node *x,node *&rt)
{
while (x!=rt)
{
node *y=x->f,*z=y->f;
if (y==rt) if (x==y->l) right(x,rt); else left(x,rt);
else if (y==z->l) if (x==y->l) {right(y,rt);right(x,rt);} else {left(x,rt);right(x,rt);}
else if (x==y->r) {left(y,rt);left(x,rt);} else {right(x,rt);left(x,rt);}
}
}
void find(int rk,node *&rt)
{
node *x=rt;
while ((x->l?x->l->sz+1:1)!=rk)
{
if (rk<=(x->l?x->l->sz:0)) x=x->l; else
{
rk-=(x->l?x->l->sz+1:1);
x=x->r;
}
}
splay(x,rt);
}
void visit(node *rt)
{
if (rt->l) visit(rt->l);
if (~rt->val) printf("%d ",rt->val);
if (rt->r) visit(rt->r);
}
public:
Splay_tree()
{
node *x=new node;
node *y=new node;
x->r=y; y->f=x; x->sz=2;
rt=x;
}
~Splay_tree(){del(rt);}
void insert(int rk,int val)
{
find(rk,rt);
find(1,rt->r);
rt->r->l=new node(val,1,rt->r);
pushup(rt->r); pushup(rt);
}
void print(){visit(rt);printf("\n");}
};
int main()
{
int n;
while (~scanf("%d",&n))
{
Splay_tree T;
for (int i=1;i<=n;i++)
{
int x,y;
scanf("%d%d",&x,&y);
T.insert(x+1,y);
}
T.print();
}
}

线段树

#include <cstdio>
#include <iostream>
using namespace std;
const int N=200008;
class Segment_tree
{
public:
struct node
{
int val,space;
}a[N*4];
void pushup(int rt)
{
a[rt].space=a[rt<<1].space+a[rt<<1|1].space;
}
void build(int l,int r,int rt)
{
a[rt].val=a[rt].space=0;
if (l==r)
{
a[rt].val=0;
a[rt].space=1;
return;
}
int m=l+r>>1;
build(l,m,rt<<1);
build(m+1,r,rt<<1|1);
pushup(rt);
}
void update(int k,int val,int l,int r,int rt)
{
if (l==r)
{
a[rt].space=0;
a[rt].val=val;
return;
}
int m=l+r>>1;
if (k<=a[rt<<1].space) update(k,val,l,m,rt<<1);
else update(k-a[rt<<1].space,val,m+1,r,rt<<1|1);
pushup(rt);
}
void query(int l,int r,int rt)
{
if (l==r)
{
printf("%d ",a[rt].val);
return;
}
int m=l+r>>1;
query(l,m,rt<<1);
query(m+1,r,rt<<1|1);
}
}T;
int x[N],y[N];
int main()
{
int n;
while (~scanf("%d",&n))
{
T.build(1,n,1);
for (int i=1;i<=n;i++) scanf("%d%d",&x[i],&y[i]);
for (int i=n;i>=1;i--) T.update(x[i]+1,y[i],1,n,1);
T.query(1,n,1); printf("\n");
}
}

树状数组

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
const int N=200008;
class Binary_index_tree
{
public:
int a[N];
void init(){memset(a,0,sizeof(a));}
void add(int x,int y)
{
for (int i=x;i<N;i+=i & (-i)) a[i]+=y;
}
int sigma(int x)
{
int res=0;
for (int i=x;i;i-=i & (-i)) res+=a[i];
return res;
}
}T;
int x[N],y[N],ans[N];
int main()
{
int n;
while (~scanf("%d",&n))
{
for (int i=1;i<=n;i++) scanf("%d%d",&x[i],&y[i]);
T.init();
for (int i=1;i<=n;i++) T.add(i,1);
for (int i=n;i>=1;i--)
{
int l=1,r=n,res=-1;
while (l<=r)
{
int m=l+r>>1;
int num=T.sigma(m);
if (num==x[i]+1) res=m;
if (num<x[i]+1) l=m+1; else r=m-1;
}
ans[res]=y[i];
T.add(res,-1);
}
for (int i=1;i<=n;i++) printf("%d ",ans[i]); printf("\n");
}
}

POJ 2828 Buy Tickets (线段树 || 树状数组)的更多相关文章

  1. poj 2828 Buy Tickets (线段树(排队插入后输出序列))

    http://poj.org/problem?id=2828 Buy Tickets Time Limit: 4000MS   Memory Limit: 65536K Total Submissio ...

  2. POJ 2828 Buy Tickets (线段树 or 树状数组+二分)

    题目链接:http://poj.org/problem?id=2828 题意就是给你n个人,然后每个人按顺序插队,问你最终的顺序是怎么样的. 反过来做就很容易了,从最后一个人开始推,最后一个人位置很容 ...

  3. POJ 2828 Buy Tickets 线段树 倒序插入 节点空位预留(思路巧妙)

    Buy Tickets Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 19725   Accepted: 9756 Desc ...

  4. poj 2828 Buy Tickets (线段树)

    题目:http://poj.org/problem?id=2828 题意:有n个人插队,给定插队的先后顺序和插在哪个位置还有每个人的val,求插队结束后队伍各位置的val. 线段树里比较简单的题目了, ...

  5. POJ 2828 Buy Tickets | 线段树的喵用

    题意: 给你n次插队操作,每次两个数,pos,w,意为在pos后插入一个权值为w的数; 最后输出1~n的权值 题解: 首先可以发现,最后一次插入的位置是准确的位置 所以这个就变成了若干个子问题, 所以 ...

  6. POJ 2828 Buy Tickets(线段树&#183;插队)

    题意  n个人排队  每一个人都有个属性值  依次输入n个pos[i]  val[i]  表示第i个人直接插到当前第pos[i]个人后面  他的属性值为val[i]  要求最后依次输出队中各个人的属性 ...

  7. POJ 2828 Buy Tickets(线段树单点)

    https://vjudge.net/problem/POJ-2828 题目意思:有n个数,进行n次操作,每次操作有两个数pos, ans.pos的意思是把ans放到第pos 位置的后面,pos后面的 ...

  8. poj 2828 Buy Tickets(树状数组 | 线段树)

    题目链接:poj 2828 Buy Tickets 题目大意:给定N,表示有个人,给定每一个人站入的位置,以及这个人的权值,如今按队列的顺序输出每一个人的权值. 解题思路:第K大元素,非常巧妙,将人入 ...

  9. POJ 2828 Buy Tickets(排队问题,线段树应用)

    POJ 2828 Buy Tickets(排队问题,线段树应用) ACM 题目地址:POJ 2828 Buy Tickets 题意:  排队买票时候插队.  给出一些数对,分别代表某个人的想要插入的位 ...

  10. poj 2828 Buy Tickets 【线段树点更新】

    题目:id=2828" target="_blank">poj 2828 Buy Tickets 题意:有n个人排队,每一个人有一个价值和要插的位置,然后当要插的位 ...

随机推荐

  1. 12.1Java-构造方法

    一.构造方法 作用:在new的同时对成员变量赋值,给对象的属性初始化赋值格式:权限 方法名(参数列表){}方法的名字,必须和类的名字完全一致,大小写一致构造方法不许写返回值类型,如void,int 构 ...

  2. JDK11源码分析之集合类(一)----HashMap

    一,首先需要拉取JDK11源码: 方便起见我给出芋道源码作者已经拉取好的openJDK11的GitHub地址只需要fork一下克隆到本地导入IDEA中就可以对源码分析了: https://github ...

  3. html5表单新增元素与属性2

    1.标签的control属性 在html5中,可以在标签内部放置一个表单元素,并且通过该标签的control属性来访问该表单元素. <script> function setValue() ...

  4. Program received signal SIGILL, Illegal instruction

    Program received signal SIGILL, Illegal instruction 这个错误,发现是直接在printf 的%s中直接使用string类型,而没有使用c字符串格式造成 ...

  5. Android 使用Bitmap将自身保存为文件,BitmapFactory从File中解析图片并防止OOM

    1.使用Bitmap将自身保存为文件 public boolean saveBitmapAsFile(String name, Bitmap bitmap) { File saveFile = new ...

  6. React 实践心得:react-redux 之 connect 方法详解

    Redux 是「React 全家桶」中极为重要的一员,它试图为 React 应用提供「可预测化的状态管理」机制. Redux 本身足够简单,除了 React,它还能够支持其他界面框架.所以如果要将 R ...

  7. OpenGl之旅-—初识opengl

    昨天学习了如何使用codeblocks来编译运行一个opengl的项目.在创建一个新的opengl项目时他默认已经写了一个示例,今天我们就上面的例子进行下代码的剖析,以此来敲开opengl的神秘大门. ...

  8. Google浏览器开发者工具:CSSViewer(一个Css查看器)

    CSSViewer的简介 CSSViewer是一款可以帮助用户快速查看当前的网页元素的CSS属性的谷歌浏览器插件,在Chrome中安装了CSSViewer插件以后,用户就可以在设计网页的时候,快速地模 ...

  9. mysql中 for update 使用

    解释: for update是在数据库中上锁用的,可以为数据库中的行上一个排它锁.当一个事务的操作未完成时候,其他事务可以读取但是不能写入或更新.例子: 比如一张表三个字段 , id(商品id), n ...

  10. SQL GROUP BY 和 ORDER BY 区别

    order by 是按表中某字段排列表中数据group by 是按某些字段分类. 例如按 1.按年龄排序表中的记录select * from users order by age 2.按年龄分类表中数 ...