今天学习了一下spaly。。。。。

感觉除了比较难打,比较难调,但还是很好理解的啊。。。。

1588: [HNOI2002]营业额统计

Time Limit: 5 Sec  Memory Limit: 162 MB
Submit: 6923  Solved: 2286
[Submit][Status]

Description

营业额统计 Tiger最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况。 Tiger拿出了公司的账本,账本上记录了公司成立以来每天的营业额。分析营业情况是一项相当复杂的工作。由于节假日,大减价或者是其他情况的时候,营业额会出现一定的波动,当然一定的波动是能够接受的,但是在某些时候营业额突变得很高或是很低,这就证明公司此时的经营状况出现了问题。经济管理学上定义了一种最小波动值来衡量这种情况: 该天的最小波动值 当最小波动值越大时,就说明营业情况越不稳定。 而分析整个公司的从成立到现在营业情况是否稳定,只需要把每一天的最小波动值加起来就可以了。你的任务就是编写一个程序帮助Tiger来计算这一个值。 第一天的最小波动值为第一天的营业额。  输入输出要求

Input

第一行为正整数 ,表示该公司从成立一直到现在的天数,接下来的n行每行有一个正整数 ,表示第i天公司的营业额。

Output

输出文件仅有一个正整数,即Sigma(每天最小的波动值) 。结果小于2^31 。

Sample Input

6
5
1
2
5
4
6

Sample Output

12

HINT

结果说明:5+|1-5|+|2-1|+|5-5|+|4-5|+|6-5|=5+4+1+0+1+1=12


这是一道splay的裸题呢。splay的题目还是很好看出来的啦。

  这道题目需要基本操作(左旋,右旋,伸展)以及寻找前驱(左子树的最大值)和后继(右子树的最小值),插入操作(insert)。

AC代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstdlib>
#include<cstring>
#include<cmath>
#define ll long long
#define maxn 100010
ll father[maxn],rightson[maxn],leftson[maxn],root,n,m,val[maxn],zhi,ds,wz,ans;
using namespace std;
ll ab(ll x){if (x<0) x=-x; return x;}
ll min(ll x,ll y)
{
if (x<y) return x;else return y;
}
void right_rotate(ll x)//右旋
{
ll y=father[x],z=father[y];
leftson[y]=rightson[x];
if (rightson[x]!=0) father[rightson[x]]=y;//处理x的右子树
father[x]=z;
if (z!=0)
{
if (leftson[z]==y) leftson[z]=x;else rightson[z]=x;
}
rightson[x]=y; father[y]=x;
}
/*
右旋只需要三步:
1.将X的右子树B(如果有的话)作为Y的左子树,同时让B认Y作爹
2.设Z为原本Y结点的父亲,让X认Z做爹(如果Z存在的话),将X作为Z的儿子(是左是右得由Y是Z的左儿子还是右儿子决定,要左右一致)
3.将Y作为X的右子树,同时让Y认X作爹
*/
void left_rotate(ll x)
{
ll y=father[x],z=father[y];
rightson[y]=leftson[x];
if(leftson[x]!=0)father[leftson[x]]=y;
father[x]=z;
if(z!=0)
{
if(leftson[z]==y)leftson[z]=x;else rightson[z]=x;
}
leftson[x]=y;father[y]=x;
}
void splay(ll x,ll llg)//表示将x不停向上旋转,知道X成为结点为llg的子树
{
while(father[x]!=llg)
{
long y=father[x];long z=father[y];
if(z==llg)
{
if(rightson[y]==x)left_rotate(x);else right_rotate(x);
}
else
{
if(rightson[z]==y && rightson[y]==x){left_rotate(y); left_rotate(x);}
else if(rightson[z]==y && leftson[y]==x) {right_rotate(x); left_rotate(x);}
else if(leftson[z]==y && leftson[y]==x) {right_rotate(y); right_rotate(x);}
else {left_rotate(x); right_rotate(x);}
}
}
if(llg==0) root=x;
}
ll findleft(ll x)//找这个节点左子树的最大值(前驱)
{
ll left=leftson[x];
while (rightson[left]!=0) left=rightson[left];
return left;
}
ll findright(ll x)//找这个节点在右子树的最小值(后继)
{
ll right=rightson[x];
while (leftson[right]!=0) right=leftson[right];
return right;
}
void find(ll x)
{
wz=x;
if (val[ds]>val[x] && rightson[x]!=0) find(rightson[x]);
if (val[ds]<=val[x] && leftson[x]!=0) find(leftson[x]);
}
void insert()
{
find(root);
father[ds]=wz;
if (ds!=1) if (val[ds]>val[wz]) rightson[wz]=ds; else leftson[wz]=ds;
splay(ds,0);
}
int main()
{
freopen("in.in","r",stdin); freopen("out.out","w",stdout);
cin>>n;
//memset(val,0x7fffffff,sizeof(val));
val[0]=0x7fffffff;
m=n; root=0;
while (m--)
{
cin>>zhi;
ds++;
val[ds]=zhi;
insert();
if (findright(ds)==0 && findleft(ds)==0) ans+=zhi; else ans+=min(ab(val[findright(ds)]-zhi),ab(val[findleft(ds)]-zhi));
}
cout<<ans;
return 0;
}

splay学习的更多相关文章

  1. 平衡树splay学习笔记#1

    这一篇博客只讲splay的前一部分的操作(rotate和splay),后面的一段博客咕咕一段时间 后一半的博客地址:[传送门] 前言骚话 为了学lct我也是拼了,看了十几篇博客,学了将近有一周,才A掉 ...

  2. 平衡树splay学习笔记#2

    讲一下另外的所有操作(指的是普通平衡树中的其他操作) 前一篇的学习笔记连接:[传送门],结尾会带上完整的代码. 操作1,pushup操作 之前学习过线段树,都知道子节点的信息需要更新到父亲节点上. 因 ...

  3. 文艺平衡树 Splay 学习笔记(1)

    (这里是Splay基础操作,reserve什么的会在下一篇里面讲) 好久之前就说要学Splay了,结果苟到现在才学习. 可能是最近良心发现自己实在太弱了,听数学又听不懂只好多学点不要脑子的数据结构. ...

  4. [Splay][学习笔记]

    胡扯 因为先学习的treap,而splay与treap中有许多共性,所以会有很多地方不会讲的很细致.关于treap和平衡树可以参考这篇博客 关于splay splay,又叫伸展树,是一种二叉排序树,它 ...

  5. $splay$学习总结$QwQ$

    省选之前就大概搞了下$splay$,然后因为时间不太够就没写总结了,,,然后太久没用之后现在一回想感觉跟没学过一样了嘤嘤嘤 所以写个简陋的总结,,,肥肠简陋,只适合$gql$复习用,不建议学习用 然后 ...

  6. [Note]Splay学习笔记

    开个坑记录一下学习Splay的历程. Code 感谢rqy巨佬的代码,让我意识到了Splay可以有多短,以及我之前的Splay有多么的丑... int fa[N], ch[N][2], rev[N], ...

  7. splay学习笔记

    伸展树(Splay Tree),也叫分裂树,是一种二叉排序树,它能在O(log n)内完成插入.查找和删除操作.(来自百科) 伸展树的操作主要是 –rotate(x) 将x旋转到x的父亲的位置 voi ...

  8. 【洛谷P3391】文艺平衡树——Splay学习笔记(二)

    题目链接 Splay基础操作 \(Splay\)上的区间翻转 首先,这里的\(Splay\)维护的是一个序列的顺序,每个结点即为序列中的一个数,序列的顺序即为\(Splay\)的中序遍历 那么如何实现 ...

  9. 【洛谷P3369】普通平衡树——Splay学习笔记(一)

    二叉搜索树(二叉排序树) 概念:一棵树,若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值: 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值: 它的左.右子树也分别为二叉搜索树 ...

随机推荐

  1. access数据库导入Oracle

    1.对着当前的表右击->导出->选择下面的保存类型为"ODBC数据库"找一个路径输入文件名2.将x导出到x,点击->确定3.在弹出的对话框中DSN名称,点击-&g ...

  2. oracle 自动关闭 数据库连接

    数据库启动后过一会自动断开连接(大概10分钟)    方法一.直接修改资源配置文件     分三个步骤在sqlplus环境下完成.     第一步,查询资源文件,找到CONNECT_TIME所在的pr ...

  3. 【jQuery UI 1.8 The User Interface Library for jQuery】.学习笔记.8.Datepicker控件

    默认datepicker的安装启用 探索它的配置选项 安装启用一个触发按钮 配置一个供选择的动画 dateFormat选项 简单的国际化 多月datepicker 日期范围选择 datepicker的 ...

  4. 今天 同一个Nav 左右button 替换不显示的问题 viewDidLoad, viewWillDisappear, viewWillAppear等区别及各自的加载顺序

    viewWillAppear:  Called when the view is about to made visible. Default does nothing视图即将可见时调用.默认情况下不 ...

  5. PHP上传文件详解 错误提示

    首先在php.ini里配置上载文件.有以下几个重要的配置单: 选项 默认值 说明 post_max_size 8M 控制以后的POST请求的最大规模.必须大于upload_max_filesize选项 ...

  6. 转Class.forName()用法详解

    主要功能 Class.forName(xxx.xx.xx)返回的是一个类 Class.forName(xxx.xx.xx)的作用是要求JVM查找并加载指定的类, 也就是说JVM会执行该类的静态代码段 ...

  7. java代码实现rabbitMQ请求

    1.下载rabbitMQ的客户端,下载地址 http://www.rabbitmq.com/download.html

  8. 搭建LAMP

    RPM包和源码包存放位置 /usr/local/src 源码包编译安装位置 /usr/local/apache /usr/local/mysql /usr/local/php 默认MySQL 数据库位 ...

  9. RecycleView使用的那些坑

    1.为条目设置margin值时,在6.0系统上会无效.此时在item的根外面套一层viewgroup解决. 2.当条目中有imageview时,必须给imageview设置 src或者backgrou ...

  10. $q服务的API详解

    下面我们通过讲解$q的API让你更多的了解promise异步编程模式.$q是做为angularjs的一个服务而存在的,只是对promise异步编程模式的一个简化实现版,源码中剔除注释实现代码也就二百多 ...