题目描述

原题来自:HNOI 2002

Tiger 最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况。

Tiger 拿出了公司的账本,账本上记录了公司成立以来每天的营业额。分析营业情况是一项相当复杂的工作。由于节假日,大减价或者是其他情况的时候,营业额会出现一定的波动,当然一定的波动是能够接受的,但是在某些时候营业额突变得很高或是很低,这就证明公司此时的经营状况出现了问题。

经济管理学上定义了一种最小波动值来衡量这种情况:记该天以前某一天的营业额为ai,该天营业额为 b,则该天的最小波动值min | ai-b | ,当最小波动值越大时,就说明营业情况越不稳定。而分析整个公司的从成立到现在营业情况是否稳定,只需要把每一天的最小波动值加起来就可以了。

你的任务就是编写一个程序帮助 Tiger 来计算这一个值,第一天的最小波动值为第一天的营业额。

一句话题意

给出一个n个数的数列{an},对于第 i 个元素ai ,定义fi =min | ai - ai|,其中1<=j<=i,fi=a1 。求 Σfi

输入格式

第一行为正整数,表示该公司从成立一直到现在的天数;

接下来的n行每行有一个正整数,表示第 i 天公司的营业额ai 。

输出格式

仅有一个正整数,即每一天最小波动的和,结果不超过1<<31 。

样例

样例输入

6
5
1
2
5
4
6

样例输出

12

样例说明

数据范围与提示

对于全部数据,1<=n<=215 ,ai<=106

_________________________________

平衡树中查找前驱后继,然后和当前值取差,取较小的加入最终结果。

继续熟悉FHQ_TREAP

_________________________________

 1 #include<bits/stdc++.h>
2 using namespace std;
3 const int maxn=(1<<15)+10;
4 const int inf=1e7;
5 int n;
6 struct node
7 {
8 int lc,rc,val,rd,siz;
9 }tr[maxn];
10 int cnt,root;
11 long long ans;
12 int newnode(int v)
13 {
14 ++cnt;
15 tr[cnt].lc=tr[cnt].rc=0;
16 tr[cnt].val=v;
17 tr[cnt].rd=rand();
18 tr[cnt].siz=1;
19 return cnt;
20 }
21 void update(int x)
22 {
23 tr[x].siz=tr[tr[x].lc].siz+tr[tr[x].rc].siz+1;
24 }
25 int merge(int x,int y)
26 {
27 if(x*y==0)return x+y;
28 if(tr[x].rd<tr[y].rd)
29 {
30 tr[x].rc=merge(tr[x].rc,y);
31 update(x);
32 return x;
33 }
34 else
35 {
36 tr[y].lc=merge(x,tr[y].lc);
37 update(y);
38 return y;
39 }
40 }
41 void split(int cur,int k,int &x,int &y)
42 {
43 if(!cur)x=y=0;
44 else
45 {
46 if(tr[cur].val<=k)
47 {
48 x=cur;
49 split(tr[cur].rc,k,tr[cur].rc,y);
50 }
51 else
52 {
53 y=cur;
54 split(tr[cur].lc,k,x,tr[cur].lc);
55 }
56 update(cur);
57 }
58 }
59 int kth(int now,int v)
60 {
61 int cur=now;
62 while(cur)
63 {
64 if(tr[tr[cur].lc].siz+1==v)return tr[cur].val;
65 else if(tr[tr[cur].lc].siz>=v)cur=tr[cur].lc;
66 else v-=tr[tr[cur].lc].siz+1,cur=tr[cur].rc;
67 }
68 return inf;
69 }
70 void work(int v)
71 {
72 int x,y,tp=inf,tpp;
73 split(root,v,x,y);
74 tpp=kth(x,tr[x].siz);
75 tp=tpp==inf?inf:v-tpp;
76 tpp=kth(y,1);
77 tp=tpp==inf?tp:min(tp,tpp-v);
78 if(tp!=inf)ans+=tp;
79 root=merge(merge(x,newnode(v)),y);
80 }
81 int main()
82 {
83 scanf("%d",&n);
84 for(int x,i=0;i<n;++i)
85 {
86 scanf("%d",&x);
87 if(i==0)ans=x;
88 work(x);
89 }
90 cout<<ans;
91 return 0;
92 }

LOJ10043的更多相关文章

随机推荐

  1. javascript中如何截取字符串?

    JavaScript中截取字符串有三种方法,分别是substring(),substr(),split(). 方法1:使用substring() substring()方法用于提取字符串中介于两个指定 ...

  2. .NET Core学习笔记(8)——Entity Framework Core之Database First

    曾经我以为再也不会去弄啥Database First,然鹅我错了.这个世界上就是有啪啪打脸和真香的时候.当小伙伴拿着做好的DB表结构和SQL脚本递过来的时候,我知道我没法拒绝.望着他突起的肱二头肌和充 ...

  3. Centos7无网络下安装mysql5.7——mysql-rpm安装

    本教程指将mysql安装到系统默认目录下,如想自定义修改目录,请在rpm安装时自行修改: rpm -ivh --prefix= /opt xxx.rpm #将xxx.rpm安装到/opt下 一.下载m ...

  4. lambda表达式之方法引用

    /** * 方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器.<br> * 与lambda联合使用,方法引用可以使语言的构造更紧凑简洁,减少冗余代码. ...

  5. 第十一章节 BJROBOT PS3 手柄控制【ROS全开源阿克曼转向智能网联无人驾驶车】

    1.把小车架空平放在地板上.   2.用 USB 线将 PS3 蓝牙手柄连接至小车主控端,初次连接手柄上的 4 个红色指示灯会同时闪烁; 3.按下手柄中间的圆形配对键,然后等待红灯闪烁至停止. 4.此 ...

  6. Git之pull,fetch差别

    简言之, pull=fetch+merge,下拉远程分支并与本地分支合并. fetch只是下拉远程分支,怎么合并,可以自己再做选择. 进一步了解是,git本地有暂存区(亦称为Index区) fetch ...

  7. #3使用html+css+js制作网页 番外篇 使用python flask 框架 (I)

    #3使用html+css+js制作网页 番外篇 使用python flask 框架(I 第一部) 0. 本系列教程 1. 准备 a.python b. flask c. flask 环境安装 d. f ...

  8. SpringCloud Gateway快速入门

    SpringCloud Gateway cloud笔记第一部分 cloud笔记第二部分Hystrix 文章目录 SpringCloud Gateway Zull的工作模式与Gateway的对比 Rou ...

  9. pidof

    pidof 服务名称,就可以查看到服务占用的进程号

  10. 【Linux】java.io.IOException: error=24, Too many open files解决

    linux系统中执行java程序的时候,如果打开文件超过了限制,就会报错: java.io.IOException: error=24, Too many open files 解决办法: 首先查看j ...