POJ 2828 Buy Tickets (线段树 || 树状数组)
题目大意
一些小朋友在排队,每次来一个人,第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 (线段树 || 树状数组)的更多相关文章
- poj 2828 Buy Tickets (线段树(排队插入后输出序列))
http://poj.org/problem?id=2828 Buy Tickets Time Limit: 4000MS Memory Limit: 65536K Total Submissio ...
- POJ 2828 Buy Tickets (线段树 or 树状数组+二分)
题目链接:http://poj.org/problem?id=2828 题意就是给你n个人,然后每个人按顺序插队,问你最终的顺序是怎么样的. 反过来做就很容易了,从最后一个人开始推,最后一个人位置很容 ...
- POJ 2828 Buy Tickets 线段树 倒序插入 节点空位预留(思路巧妙)
Buy Tickets Time Limit: 4000MS Memory Limit: 65536K Total Submissions: 19725 Accepted: 9756 Desc ...
- poj 2828 Buy Tickets (线段树)
题目:http://poj.org/problem?id=2828 题意:有n个人插队,给定插队的先后顺序和插在哪个位置还有每个人的val,求插队结束后队伍各位置的val. 线段树里比较简单的题目了, ...
- POJ 2828 Buy Tickets | 线段树的喵用
题意: 给你n次插队操作,每次两个数,pos,w,意为在pos后插入一个权值为w的数; 最后输出1~n的权值 题解: 首先可以发现,最后一次插入的位置是准确的位置 所以这个就变成了若干个子问题, 所以 ...
- POJ 2828 Buy Tickets(线段树·插队)
题意 n个人排队 每一个人都有个属性值 依次输入n个pos[i] val[i] 表示第i个人直接插到当前第pos[i]个人后面 他的属性值为val[i] 要求最后依次输出队中各个人的属性 ...
- POJ 2828 Buy Tickets(线段树单点)
https://vjudge.net/problem/POJ-2828 题目意思:有n个数,进行n次操作,每次操作有两个数pos, ans.pos的意思是把ans放到第pos 位置的后面,pos后面的 ...
- poj 2828 Buy Tickets(树状数组 | 线段树)
题目链接:poj 2828 Buy Tickets 题目大意:给定N,表示有个人,给定每一个人站入的位置,以及这个人的权值,如今按队列的顺序输出每一个人的权值. 解题思路:第K大元素,非常巧妙,将人入 ...
- POJ 2828 Buy Tickets(排队问题,线段树应用)
POJ 2828 Buy Tickets(排队问题,线段树应用) ACM 题目地址:POJ 2828 Buy Tickets 题意: 排队买票时候插队. 给出一些数对,分别代表某个人的想要插入的位 ...
- poj 2828 Buy Tickets 【线段树点更新】
题目:id=2828" target="_blank">poj 2828 Buy Tickets 题意:有n个人排队,每一个人有一个价值和要插的位置,然后当要插的位 ...
随机推荐
- Vmware workstation12里如何正确快速安装可视化IDS系统Security Onion(图文详解)
不多说,直接上干货! 首先,大家要明确: 问:安全洋葱能阻止入侵吗? 答:这一点,和OSSIM一样,不能阻止入侵. Security Onion基于Ubuntu,包含了入侵检测.网络安全监控.日志管理 ...
- [转]如何在 TFS 中使用 Git
本文转自 http://www.cnblogs.com/stg609/p/3651688.html 对 Charley Blog 的代码进行版本控制的想法由来已久,在代码建立之初其实已经使用过 TFS ...
- redis 远程连接方法
解决方法 1.修改redis服务器的配置文件 vi redis.conf 注释以下绑定的主机地址 # bind 127.0.0.1 或 vim redis.conf bind 0.0.0.0 pr ...
- 2017.5.20欢(bei)乐(ju)赛解题报告
预计分数:100+20+50=first 实际分数:20+0+10=gg 水灾(sliker.cpp/c/pas) 1000MS 64MB 大雨应经下了几天雨,却还是没有停的样子.土豪CCY刚从外地 ...
- 12 DOM操作应用
1.创建子元素oLi=document.creatElement('li') 2.将元素附给父级元素oUl.appendChild(oLi) 3.将元素插入到父级元素里的第一位子元素之前oUl.ins ...
- mvc框架 与vuex的介绍
应用级的状态集中放在store中: 改变状态的方式是提交mutations,这是个同步的事物: 异步逻辑应该封装在action中. const vuex_store = new Vuex.store( ...
- CREATE GROUP - 定义一个新的用户组
SYNOPSIS CREATE GROUP name [ [ WITH ] option [ ... ] ] where option can be: SYSID gid | USER usernam ...
- COMMENT - 定义或者改变一个对象的评注
SYNOPSIS COMMENT ON { TABLE object_name | COLUMN table_name.column_name | AGGREGATE agg_name (agg_ty ...
- mfc 菜单
创建一个基于对话框的工程,工程名为CreateMenu 为该对话框增加一个文件菜单项和测试菜单项,如下图所示 测试菜单项至少要有一个子菜单项 在对话框属性中关联该菜单 在resource.h中增加 ...
- vue工程化与路由router
一.介绍 vue.js 是 目前 最火的前端框架,vue.js 兼具 angular.js 和 react.js 的优点,并剔除它们的缺点.并且提供了很多的周边配套工具 如vue-router ...