左偏树(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| ...
随机推荐
- If no other git process is currently running, this probably means a git proc
原因:用SourceTree提交代码,发现这个问题.好像是因为上个进程没停止,造成文件不识别 解决:把仓库目录里的.git/index.lock文件(文件是隐藏的)删除就可以了.删除index.loc ...
- Bootstrap框架中radio设置值
Bootstrap中的radio设置值不能像我们平常给普通radio赋值那样,因为无效. 我们用Bootstrap框架里的radio组件,代码: <div class="radio-l ...
- 题解 CF409A 【The Great Game】
题目传送门. 思路: 首先我们定义\(2\)个字符串,分别存放 TEAM 1 与 TEAM 2 的出招顺序.接着再定义\(2\)个变量,存放 TEAM 1 与 TEAM 2 的分数. string s ...
- Python标准库之时间模块time与datatime模块详解
时间模块time与datatime 时间表示方式: 时间戳 格式化时间字符串 元组 时间戳格式: time.time()#输出1581664531.749063 元组格式: time.localtim ...
- C语言strcat()函数:字符串连接(拼接)
C语言strcat()函数:字符串连接(拼接) C语言 strcat() 函数用来将两个字符串连接(拼接)起来. 头文件:string.h 语法/原型: char*strcat(char* str ...
- ubuntu 报错 Unable to locate package
解决方案参考 https://blog.csdn.net/u010622613/article/details/83017163
- NG-ALAIN 边学边记1
在文件夹下右键启动powerShell ng new my-project --skip-npm cd my-project ng add ng-alainnpm installng serve np ...
- 服务器上搭建使用SSH账户登录的Git仓库
1.安装git yum install -y git 2.创建git仓库保存的目录 mkdir /data/git_repo 3.初始化空仓库 cd /data/git_repogit init -- ...
- http接口性能测试工具
一. http_load 程序非常小,解压后也不到100K.http_load以并行复用的方式运行,用以测试web服务器的吞吐量与负载. 但是它不同于大多数压力测试工具,它可以以一个单一的进程运行,一 ...
- Activiti+Shiro实战
有人曾说:人的差距都在业余时间拉开的……嗯,我现在深刻理解着这句话,作为一个程序员,技术男,就得不断学习新的技术,跟上时代步伐,才会让自己更有价值~~~~以下这个项目是个人利用业余时间学习并实践的~如 ...