题面:BZOJ传送门

题目大意:给你一个序列$a$,让你构造一个递增序列$b$,使得$\sum |a_{i}-b_{i}|$最小,$a_{i},b_{i}$均为整数

神仙题..

我们先考虑b不递减的情况

假设现在有一段单调的序列$A$

如果$A$是递增的,显然$b[i]=a[i]$是最优解

如果$A$是递减的,$b$的每一项=序列$A$的中位数时是最优解

简单证明一下递减的情况:

1.序列$A$元素数量是奇数时,我们以中位数为对称轴,那么对称的两个数带来的贡献就是它们的差值,而中位数本身不会产生贡献,如果选取的不是中位数,必然会导致中位数产生贡献,而且对称的两个数还可能产生差值以外的贡献

2.序列$A$元素数量是偶数时,选取的数在中间的两个数之间即可,贡献都是一样的

我们如何利用这一性质呢?

我们把序列拆分成很多递减的段,递增子段的每一个数都是单独的一段

我们的目的是保证$b$不递减,即把每一段取的数画成一个函数来看是不递减的

每次我们在序列末尾加入一个数$a_{i}$,都看看这一段的中位数是否$\geq $前面一段的中位数,不满足就和前一段合并,然后依次重复此过程

为什么答案合并后不会变差?太弱了并不会证..感性理解一下吧。因为这两段原来取的就是最优解,但并不满足要求,我们也只能把两段合并,再用相同的方法求最优解了..

具体实现可以用左偏树维护大根堆,记录每一段较大的$\left \lceil \frac{n}{2} \right \rceil$个数,当两个奇数段合并时删除一次堆顶即可,记录每一段的信息可以用结构体

 #include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 1000010
#define ll long long
using namespace std;
const ll inf=0x3f3f3f3f3f3f3f3fll; template <typename _T> void read(_T &ret)
{
ret=; _T fh=; char c=getchar();
while(c<''||c>''){ if(c=='-') fh=-; c=getchar(); }
while(c>=''&&c<=''){ ret=ret*+c-''; c=getchar(); }
ret=ret*fh;
} struct Heap{
int fa[N1],ch[N1][],h[N1]; ll val[N1];
int merge(int x,int y)
{
if(!x||!y) return x+y;
if(val[x]<val[y]) swap(x,y);
//pushdown(x);
ch[x][]=merge(ch[x][],y); fa[ch[x][]]=x;
if(h[ch[x][]]<h[ch[x][]]) swap(ch[x][],ch[x][]);
h[x]=h[ch[x][]]+;
return x;
}
int del(int x)
{
fa[ch[x][]]=fa[ch[x][]]=;
int y=merge(ch[x][],ch[x][]);
ch[x][]=ch[x][]=;
return y;
}
}h; ll a[N1];
struct node{int l,r,x;};
node stk[N1]; int n,tp; int main()
{
scanf("%d",&n);
int i,j,x,y,len; node K; ll ans=,tmp;
for(i=;i<=n;i++) read(a[i]);
for(i=;i<=n;i++)
{
h.val[i]=a[i]-i; x=i; len=;
while(tp)
{
K=stk[tp];
if(h.val[x]>=h.val[K.x]) break;
x=h.merge(x,K.x); tp--;
if( (len&) && ((K.r-K.l+)&) ) x=h.del(x);
len+=K.r-K.l+;
}
stk[++tp]=(node){i-len+,i,x};
}
for(i=;i<=tp;i++)
for(j=stk[i].l;j<=stk[i].r;j++)
{
tmp=a[j]-(h.val[stk[i].x]+j);
ans+=((tmp>)?tmp:-tmp);
}
printf("%lld\n",ans);
return ;
}

BZOJ 1367 [Baltic2004]sequence (可并堆)的更多相关文章

  1. BZOJ 1367: [Baltic2004]sequence [可并堆 中位数]

    1367: [Baltic2004]sequence Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 1111  Solved: 439[Submit][ ...

  2. BZOJ 1367 [Baltic2004]sequence 解题报告

    BZOJ 1367 [Baltic2004]sequence Description 给定一个序列\(t_1,t_2,\dots,t_N\),求一个递增序列\(z_1<z_2<\dots& ...

  3. bzoj 1367: [Baltic2004]sequence

    1367: [Baltic2004]sequence Time Limit: 20 Sec  Memory Limit: 64 MB Description Input Output 一个整数R Sa ...

  4. 【BZOJ 1367】 1367: [Baltic2004]sequence (可并堆-左偏树)

    1367: [Baltic2004]sequence Description Input Output 一个整数R Sample Input 7 9 4 8 20 14 15 18 Sample Ou ...

  5. BZOJ 1367([Baltic2004]sequence-左偏树+中位数贪心)

    1367: [Baltic2004]sequence Time Limit: 20 Sec   Memory Limit: 64 MB Submit: 521   Solved: 159 [ Subm ...

  6. 1367: [Baltic2004]sequence

    1367: [Baltic2004]sequence Time Limit: 20 Sec  Memory Limit: 64 MB Submit: 1090  Solved: 432 [Submit ...

  7. 【BZOJ】1367: [Baltic2004]sequence

    题意 给\(n(n \le 10^6)\)个数的序列\(a\),求一个递增序列\(b\)使得\(\sum_{i=1}^{n} |a_i-b_i|\)最小. 分析 神题啊不会. 具体证明看黄源河论文&l ...

  8. 【bzoj1367】[Baltic2004]sequence 可并堆

    题目描述 输入 输出 一个整数R 样例输入 7 9 4 8 20 14 15 18 样例输出 13 题解 可并堆,黄源河<左偏树的特点及其应用>Page 13例题原题 #include & ...

  9. 【bzoj1367】[Baltic2004]sequence

    2016-05-31 17:31:26 题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1367 题解:http://www.cnblogs.co ...

随机推荐

  1. 听说”双11”是这么解决线上bug的

    听说"双11"是这么解决线上bug的 --Android线上热修复的使用与原理 预备知识和开发环境 Android NDK编程 AndFix浅析 Android线上热修复的原理大同 ...

  2. 为什么用clojure作为storm 的主要开发语言

    Why you choose Clojure as the development language of Storm? Could you talk about your long practica ...

  3. 高速学会Mac上托管代码到github(具体解释)

    之前最開始的时候就一直在github浏览下载各种代码,然后弄了一下代码上传不知道咋弄就不了了之了.刚好近期有空余时间就研究了下github托管代码,这里就具体说说怎样高速的学会github上传你的代码 ...

  4. ROS探索总结(十九)——怎样配置机器人的导航功能

    1.概述 ROS的二维导航功能包.简单来说.就是依据输入的里程计等传感器的信息流和机器人的全局位置,通过导航算法,计算得出安全可靠的机器人速度控制指令. 可是,怎样在特定的机器人上实现导航功能包的功能 ...

  5. [Linux]RedHat Linux 忘记rootpassword该怎样又一次设置password

    1. 开机在出现grub画面,按e键,例如以下图所看到的: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvU3VubnlZb29uYQ==/font/5a6 ...

  6. BZOJ 1005 [HNOI2008]明明的烦恼 purfer序列,排列组合

    1005: [HNOI2008]明明的烦恼 Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少 ...

  7. POJ2228 Naptime 环形DP

    题目大意:牛在第i个小时睡觉能够恢复U[i]点体力.睡觉时第一小时不恢复体力.一天的N小时连着下一天的1小时.求能够恢复体力的和的最大值. 定义DP[i][j][0]为前i个小时休息了j个小时,i小时 ...

  8. 饭卡(hdoj--2546--背包)

    饭卡 Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submiss ...

  9. 【HAOI 2008】 移动玩具

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1054 [算法] 广度优先搜索 [代码] #include<bits/stdc+ ...

  10. 认识React框架

    在大厂面试的时候被问会不会React框架几乎是必须的,可见React框架在现在前端市场的份额.所以说学习React框架的必要性. react框架起源于Facebook的内部项目,因为对市场上的Java ...