左偏树(p4431)
难得不是左偏树,而是思维;
这道题在做得时候,有两个性质
1.如果a是一个不下降序列,那么b[i]==a[i]时取得最优解。
2.如果a是一个严格递减序列,则取a序列的中位数x,令b[1]=b[2]=b[3]=...=b[n]=x,即是最优解。
于是在做得时候,我们会分为几个区间,通过区间得合并去做这一道题;
我们根据这两个性质,求出这些区间的最优质,去合并;
三.考虑一般情况
a序列一定不可能这么良心是上面的两种情况。
但它一定是由这两种情况组成的,也就是把a序列看成一段一段的,每一段要么不下降,要么严格递减。
那么要分别计算出每一段的答案是很容易的。
问题是要保证b序列不下降,所以该怎么合并答案呢?
这里又有一个结论:
把两段合在一起,取一个新的中位数就行了=。=
道理是同上的。
四.具体操作
1.初始令每一段的长度为1,令中位数为ci,则ci = ai,然后一段一段的合并起来。
若ci <= ci+1,那么就保持不变;否则将ci和ci+1所在的区间合并,取一个新的中位数,作为新区间的答案。
.........................................................................................................................................
2.这里会出现一个问题,就是第一次合并时,有可能ci+1>=ci,没有把两个区间并起来取中位数。
但是可能后面的那个区间又和其他区间合并了,中位数变小了,以至于还要和前一个区间合并。
其实很简单qwq,用栈维护一下就好了。
.........................................................................................................................................
3.那么问题来了,怎么求中位数呢?求了中位数还要把两段区间合并起来?
(下面一段话引用于某dalao博客)
因此我们需要一个数据结构,支持合并、查询最大值和删除。
为什么要查询最大值和删除呢?因为维护中位数可以只维护⌈1/2区间长度⌉小的数,用一个大根堆,则堆顶就是中位数。
合并完两个区间后,就一直删除堆顶,直到元素个数 = ⌈1/2区间长度⌉。
显然是用左偏树啦qwq。(参照别人博客)
#include<cstdio>
#include<algorithm>
#include<math.h>
#include<string.h>
using namespace std;
typedef long long ll;
const ll maxn=1e6+;
ll val[maxn];
ll dis[maxn];
struct node{
ll rt,l,r,siz;
ll w;
}G[maxn]; ll num=;
ll ch[maxn][];
ll b[maxn];
ll Merge(ll x,ll y)
{
if(!x||!y) return x+y;
if(val[x]<val[y]) swap(x,y);
ch[x][]=Merge(ch[x][],y);
if(dis[ch[x][]]<dis[ch[x][]]) swap(ch[x][],ch[x][]);
dis[x]=dis[ch[x][]]+;
return x;
}
int main()
{
ll n;
scanf("%lld",&n);
for(ll i=;i<=n;i++){
scanf("%lld",&val[i]);
val[i]-=i;
}
for(ll i=;i<=n;i++){
G[++num]=(node) {i,i,i,,val[i] };
while(num>&&G[num].w<G[num-].w){
num--;
G[num].rt=Merge(G[num].rt,G[num+].rt);
G[num].siz+=G[num+].siz;
G[num].r=G[num+].r;
while(G[num].siz>(G[num].r-G[num].l++)>>){
G[num].siz--;
ll t=G[num].rt;
G[num].rt=Merge(ch[t][],ch[t][]);
}
G[num].w=val[G[num].rt];
}
}
ll ans=;
for(ll i=;i<=num;i++){
for(ll j=G[i].l;j<=G[i].r;j++){
b[j]=G[i].w;
ans+=fabs(val[j]-b[j]);
}
}
printf("%lld\n",ans);
for(ll i=;i<=n;i++){
printf("%lld ",b[i]+i);
}
return ;
}
左偏树(p4431)的更多相关文章
- BZOJ 1455 罗马游戏 ——左偏树
[题目分析] 左偏树的模板题目,大概就是尽量维护树的深度保持平衡,以及尽可能的快速合并的一种堆. 感觉和启发式合并基本相同. 其实并没有快很多. 本人的左偏树代码自带大常数,借鉴请慎重 [代码] #i ...
- 【BZOJ-1455】罗马游戏 可并堆 (左偏树)
1455: 罗马游戏 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1355 Solved: 561[Submit][Status][Discuss] ...
- 【bzoj2809】[Apio2012]dispatching 左偏树
2016-05-31 15:56:57 题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2809 直观的思想是当领导力确定时,尽量选择薪水少的- ...
- zoj 2334 Monkey King/左偏树+并查集
原题链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1389 大致题意:N只相互不认识的猴子(每只猴子有一个战斗力值) 两只 ...
- POJ3016-K-Monotonic(左偏树+DP)
我觉得我要改一下签名了……怎么会有窝这么啰嗦的人呢? 做这题需要先学习左偏树<左偏树的特点及其应用> 然后做一下POJ3666,这题的简单版. 思路: 考虑一下维护中位数的过程原数组为A, ...
- POJ3666-Making the Grade(左偏树 or DP)
左偏树 炒鸡棒的论文<左偏树的特点及其应用> 虽然题目要求比论文多了一个条件,但是……只需要求非递减就可以AC……数据好弱…… 虽然还没想明白为什么,但是应该觉得应该是这样——求非递减用大 ...
- bzoj 1455: 罗马游戏 左偏树+并查集
1455: 罗马游戏 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 668 Solved: 247[Submit][Status] Descriptio ...
- 左偏树(Leftist Heap/Tree)简介及代码
左偏树是一种常用的优先队列(堆)结构.与二叉堆相比,左偏树可以高效的实现两个堆的合并操作. 左偏树实现方便,编程复杂度低,而且有着不俗的效率表现. 它的一个常见应用就是与并查集结合使用.利用并查集确定 ...
- 黄源河《左偏树的应用》——数字序列(Baltic 2004)
这道题哪里都找不到. [问题描述] 给定一个整数序列a1, a2, … , an,求一个不下降序列b1 ≤ b2 ≤ … ≤ bn,使得数列{ai}和{bi}的各项之差的绝对值之和 |a1 - b1| ...
随机推荐
- 学习Python常用的工具
Python编程语言 Python是一门高级计算机程序设计语言! Python是一种解释型(脚本)语言,因为其代码简明,书写效率高,功能强大.易扩展.有丰富的专业库而受大众欢迎! 最常用的专业库有: ...
- Wannafly Camp 2020 Day 6N. 合并!
#include <bits/stdc++.h> using namespace std; int n,a[2005]; int main() { long long ans=0; cin ...
- spring中实现基于注解实现动态的接口限流防刷
本文将介绍在spring项目中自定义注解,借助redis实现接口的限流 自定义注解类 import java.lang.annotation.ElementType; import java.lang ...
- Jmeter-简介及安装
一.Jmeter简介 Apache Jmeter 是Apache组织的开放源代码项目,是一个纯java桌面应用,用于压力测试和性能测量.它最初被设计用于Web应用测试但后来扩展到其它测试领域. Apa ...
- liunx 中设置zookeeper 自启动(service zookeeper does not support chkconfig)
在liunx 上设置zookeeper 自启动 1.进入目录 cd /etc/init.d 2.创建一个文件 vim zookeeper 3.编辑zookeepr 文件 连接liunx使用的软件是fi ...
- uipath_excel
1.excel建表 https://jingyan.baidu.com/article/95c9d20d0ee5e2ec4e75618d.html 2.具体操作 https://blog.csdn.n ...
- url 加参数
url = 访问地址 + ? key1=value1 & key2 = value2 304(未修改)自从上次请求后,请求的网页未修改过.服务器返回此响应时,不会返回网页内容. 加参数,骗过服 ...
- os和sys模块_python
一.os模块 1.os模块的功能 提供对系统调用的借口,常用于系统文件目录打交道. 2.常用的方法 二.sys模块 1.模块功能 与python解释器交互 2.常用方法 print(sys.path) ...
- 使用acme.sh签发Let's Encrypt的免费数字证书
--------------安装----------------curl https://get.acme.sh | sh#让alias生效source ~/.bashrc ------------- ...
- 素问 - 使用 PE、PB 做估值
摘自<小韭的学习圈> Q 哪些行业用PE看合适,哪些用PB看合适啊?其中的大致逻辑是什么? A PE = 股价 / 每股收益 使用PE的逻辑是,我们认为一个股票有价值,是因为公司未来能赚钱 ...