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个人排队,每一个人有一个价值和要插的位置,然后当要插的位 ...
随机推荐
- Java_JDBC连接数据库_使用读取配置文件的方式
package com.homewoek3_4.dao; import java.io.IOException; import java.io.InputStream; import java.sql ...
- WPF学习07:MVVM 预备知识之数据绑定
MVVM是一种模式,而WPF的数据绑定机制是一种WPF内建的功能集,两者是不相关的. 但是,借助WPF各种内建功能集,如数据绑定.命令.数据模板,我们可以高效的在WPF上实现MVVM.因此,我们需要对 ...
- scala如何在任意方法中打印当前线程栈信息(StackTrace)
1.以wordcount为例 package org.apache.spark.examples import org.apache.spark.{SparkConf, SparkContext} / ...
- Objective -C Object initialization 对象初始化
Objective -C Object initialization 对象初始化 1.1 Allocating Objects 分配对象 Allocation is the process by w ...
- ALTER DOMAIN - 改变一个域的定义
SYNOPSIS ALTER DOMAIN name { SET DEFAULT expression | DROP DEFAULT } ALTER DOMAIN name { SET | DROP ...
- 解决vue项目eslint校验 Do not use 'new' for side effects 的两种方法
import Vue from 'vue' import App from './App.vue' import router from './router' new Vue({ el: '#app' ...
- windows环境开启PHP fileinfo扩展
fileinfo作用:本模块中的函数通过在文件的给定位置查找特定的 魔术 字节序列 来猜测文件的内容类型以及编码(通俗来讲就是获取文件的MIME信息) 开启PHP fileinfo扩展的方法: 1.下 ...
- CAD参数绘制直径标注(网页版)
主要用到函数说明: _DMxDrawX::DrawDimDiametric 绘制一个直径标注.详细说明如下: 参数 说明 DOUBLE dChordPointX 在被标注的曲线上的第一个点X值 DOU ...
- Duplicate fragment name ERROR Jetty Maven Plugin
http://stackoverflow.com/questions/5802096/duplicate-fragment-name-error-jetty-maven-plugin 4down vo ...
- ios远程推送和python版push server相关笔记
今天研究了下ios的远程推送,网上的相关教程很多,做了一遍下来记录一下遇到的问题和注意事项(转载请注明) 1.证书及乱七八糟的配置 公钥:app id管理那儿的“Development Push SS ...