codeforces675D
Tree Construction
During the programming classes Vasya was assigned a difficult problem. However, he doesn't know how to code and was unable to find the solution in the Internet, so he asks you to help.
You are given a sequence a, consisting of n distinct integers, that is used to construct the binary search tree. Below is the formal description of the construction process.
- First element a1 becomes the root of the tree.
- Elements a2, a3, ..., an are added one by one. To add element ai one needs to traverse the tree starting from the root and using the following rules:
- The pointer to the current node is set to the root.
- If ai is greater than the value in the current node, then its right child becomes the current node. Otherwise, the left child of the current node becomes the new current node.
- If at some point there is no required child, the new node is created, it is assigned value ai and becomes the corresponding child of the current node.
Input
The first line of the input contains a single integer n (2 ≤ n ≤ 100 000) — the length of the sequence a.
The second line contains n distinct integers ai (1 ≤ ai ≤ 109) — the sequence aitself.
Output
Output n - 1 integers. For all i > 1 print the value written in the node that is the parent of the node with value ai in it.
Examples
3
1 2 3
1 2
5
4 2 3 1 6
4 2 2 4
Note
Picture below represents the tree obtained in the first sample.
Picture below represents the tree obtained in the second sample.
sol:一个很显然的性质就是当前这个点父亲的权值一定是他的前驱后者后继,但是怎么判断是前驱的右儿子或者后继的左儿子嫩?貌似有点蛋碎,挂了两次之后知道了是出现时间最晚的那个就是了,前驱后继写(copy)一个splay就可以了
#include <bits/stdc++.h>
using namespace std;
typedef int ll;
inline ll read()
{
ll s=;
bool f=;
char ch=' ';
while(!isdigit(ch))
{
f|=(ch=='-'); ch=getchar();
}
while(isdigit(ch))
{
s=(s<<)+(s<<)+(ch^); ch=getchar();
}
return (f)?(-s):(s);
}
#define R(x) x=read()
inline void write(ll x)
{
if(x<)
{
putchar('-'); x=-x;
}
if(x<)
{
putchar(x+''); return;
}
write(x/);
putchar((x%)+'');
return;
}
#define W(x) write(x),putchar(' ')
#define Wl(x) write(x),putchar('\n')
const int N=,inf=0x3f3f3f3f;
int n,a[N],b[N];
namespace Pht
{
int Points,Root;
int Child[N][];
int Parent[N];
int Quanzhi[N];
int Cnt[N];
int Size[N]; inline void Init();
inline int Check(int x);
inline void PushUp(int x);
inline void Rotate(int x);
inline void Splay(int At,int To);
inline void Insert(int Val);
inline void Remove(int Val);
inline void Find(int Val);
inline int Ask_Lower(int Val);
inline int Ask_Upper(int Val);
inline int Ask_Kth(int Id);
inline void Solve(); inline void Init()
{
Points=Root=;
Insert(-inf);
Insert(inf);
}
inline int Check(int x)
{
return (Child[Parent[x]][]==x)?:;
}
inline void PushUp(int x)
{
Size[x]=Size[Child[x][]]+Size[Child[x][]]+Cnt[x];
}
inline void Rotate(int x)
{
int y,z,oo;
y=Parent[x];
z=Parent[y];
oo=Check(x);
Child[y][oo]=Child[x][oo^]; Parent[Child[x][oo^]]=y;
Child[z][Check(y)]=x; Parent[x]=z;
Child[x][oo^]=y; Parent[y]=x;
PushUp(x); PushUp(y);
}
inline void Splay(int At,int To)
{
while(Parent[At]!=To)
{
int Father=Parent[At];
if(Parent[Father]==To)
{
Rotate(At);
}
else if(Check(At)==Check(Father))
{
Rotate(Father); Rotate(At);
}
else
{
Rotate(At); Rotate(At);
}
}
if(To==) Root=At;
}
inline void Insert(int Val)
{
int Now=Root,Par=;
while(Now&&(Quanzhi[Now]!=Val))
{
Par=Now;
Now=Child[Now][(Val>Quanzhi[Now])?:];
}
if(Now)
{
Cnt[Now]++; Size[Now]++;
}
else
{
Now=++Points;
if(Par)
{
Child[Par][(Val>Quanzhi[Par])?:]=Now;
}
Parent[Now]=Par;
Child[Now][]=Child[Now][]=;
Cnt[Now]=Size[Now]=;
Quanzhi[Now]=Val;
}
Splay(Now,);
}
inline void Remove(int Val)
{
int Lower=Ask_Lower(Val),Upper=Ask_Upper(Val);
Splay(Lower,);
Splay(Upper,Lower);
if(Cnt[Child[Upper][]]>)
{
Cnt[Child[Upper][]]--; Size[Child[Upper][]]--; Splay(Child[Upper][],);
}
else
{
Child[Upper][]=;
}
}
inline void Find(int Val)
{
int Now=Root;
if(!Now) return;
while(Child[Now][(Val>Quanzhi[Now])?:]&&(Quanzhi[Now]!=Val))
{
Now=Child[Now][(Val>Quanzhi[Now])?:];
}
Splay(Now,);
}
inline int Ask_Lower(int Val)
{
Find(Val);
int Now=Root;
if(Quanzhi[Now]<Val) return Now;
Now=Child[Now][];
while(Child[Now][]) Now=Child[Now][];
return Now;
}
inline int Ask_Upper(int Val)
{
Find(Val);
int Now=Root;
if(Quanzhi[Now]>Val) return Now;
Now=Child[Now][];
while(Child[Now][]) Now=Child[Now][];
return Now;
}
inline int Ask_Kth(int Id)
{
int Now=Root;
if(Size[Root]-<Id) return ;
for(;;)
{
if(Size[Child[Now][]]>=Id)
{
Now=Child[Now][];
}
else if(Size[Child[Now][]]+Cnt[Now]<Id)
{
Id-=Size[Child[Now][]]+Cnt[Now];
Now=Child[Now][];
}
else return Now;
}
}
int Time[N];
inline void Solve()
{
Init();
int i,Min=a[],Max=a[],tt=;
Insert(a[]); Time[Root]=++tt;
for(i=;i<=n;i++)
{
int Lower=Ask_Lower(a[i]),Upper=Ask_Upper(a[i]);
// printf("T[%d]=%d T[%d]=%d ",Lower,Time[Lower],Upper,Time[Upper]);
if(Time[Lower]>Time[Upper])
{
W(Quanzhi[Lower]);
}
else
{
W(Quanzhi[Upper]);
}
Insert(a[i]);
Time[Root]=++tt;
// puts("");
}
}
}
int main()
{
int i;
R(n);
for(i=;i<=n;i++) R(a[i]);
Pht::Solve();
return ;
}
/*
Input
3
1 2 3
Output
1 2 Input
5
4 2 3 1 6
Output
4 2 2 4 Input
10
991309218 517452607 870021923 978357992 136426010 10601767 302627526 883615372 163475700 600546765
Output
991309218 517452607 870021923 517452607 136426010 136426010 978357992 302627526 870021923
*/
codeforces675D的更多相关文章
- Codeforces675D(SummerTrainingDay06-J)
D. Tree Construction time limit per test:2 seconds memory limit per test:256 megabytes input:standar ...
- codeforces675D Tree Construction
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
随机推荐
- HttpClient在.NET Core中的正确打开方式
问题来源 长期以来,.NET开发者都通过下面的方式发送http请求: using (var httpClient = new HttpClient()) { var response = await ...
- 使用SSL安全证书和nginx配置将域名HTTPS化
一.在阿里云后台申请免费版证书: 二.在域名解析里面添加记录: 三.提交审核: 四.等待审核通过后,下载nginx证书: 五.按照文档修改nginx配置文件: https://help.aliyun. ...
- C#通过序列化实现深表复制
利用二进制序列化的方式进行深拷贝 有一个缺陷 序列化的类型必须标识为刻序列化的[Serializable] 否则无法进行二进制序列化 class Program { static void Main ...
- c#解决TCP“粘包”问题
一:TCP粘包产生的原理 1,TCP粘包是指发送方发送的若干包数据到接收方接收时粘成一包,从接收缓冲区看,后一包数据的头紧接着前一包数据的尾.出现粘包现象的原因是多方面的,它既可能由发送方造成,也可能 ...
- 26 , CSS 构造表单
1. 表单标签使用 2. 下拉菜单背景 3. 滚动条的使用 4. 结构化表单布局 1 1 1 1. . . . 表单标签的使用 <label for=”name”>姓名: <inpu ...
- CSS中的一下小技巧1之CSS3三角形运用
使用CSS3实现三角形: 在前端页面中有很多时候会遇到需要三角形图案的时候,以前不知道可以用CSS3实现三角形的时候,一般都是叫UI把三角形图案切出来. 后来知道原来可以用CSS3实现三角形,可是用过 ...
- 从零学习Fluter(八):Flutter的四种运行模式--Debug、Release、Profile和test以及命名规范
从零学习Fluter(八):Flutter的四种运行模式--Debug.Release.Profile和test以及命名规范 好几天没有跟新我的这个系列文章,一是因为这两天我又在之前的基础上,重新认识 ...
- Git - git branch - 查看本地仓分支列表
索引: 目录索引 参看代码 GitHub: git.txt 一.示例: git branch 二.说明: 该命令将列出本地所有存在分支, 包括 本地独有分支与远端在本地签出的分支, 但是没有签出的远端 ...
- Spark MLlib FPGrowth关联规则算法
一.简介 FPGrowth算法是关联分析算法,它采取如下分治策略:将提供频繁项集的数据库压缩到一棵频繁模式树(FP-tree),但仍保留项集关联信息.在算法中使用了一种称为频繁模式树(Frequent ...
- JQuery 图片轮播,详细注释说明,让你一看就会!
准备工作: 1.准备几张大小相同的图片 完成功能: 1.自动轮播 2.手动轮播 3.点击二侧按钮前后切换图片 JQuery插件地址: 链接:https://pan.baidu.com/s/1zNl2- ...