BZOJ 1367 [Baltic2004]sequence (可并堆)
题面: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 (可并堆)的更多相关文章
- BZOJ 1367: [Baltic2004]sequence [可并堆 中位数]
1367: [Baltic2004]sequence Time Limit: 20 Sec Memory Limit: 64 MBSubmit: 1111 Solved: 439[Submit][ ...
- BZOJ 1367 [Baltic2004]sequence 解题报告
BZOJ 1367 [Baltic2004]sequence Description 给定一个序列\(t_1,t_2,\dots,t_N\),求一个递增序列\(z_1<z_2<\dots& ...
- bzoj 1367: [Baltic2004]sequence
1367: [Baltic2004]sequence Time Limit: 20 Sec Memory Limit: 64 MB Description Input Output 一个整数R Sa ...
- 【BZOJ 1367】 1367: [Baltic2004]sequence (可并堆-左偏树)
1367: [Baltic2004]sequence Description Input Output 一个整数R Sample Input 7 9 4 8 20 14 15 18 Sample Ou ...
- BZOJ 1367([Baltic2004]sequence-左偏树+中位数贪心)
1367: [Baltic2004]sequence Time Limit: 20 Sec Memory Limit: 64 MB Submit: 521 Solved: 159 [ Subm ...
- 1367: [Baltic2004]sequence
1367: [Baltic2004]sequence Time Limit: 20 Sec Memory Limit: 64 MB Submit: 1090 Solved: 432 [Submit ...
- 【BZOJ】1367: [Baltic2004]sequence
题意 给\(n(n \le 10^6)\)个数的序列\(a\),求一个递增序列\(b\)使得\(\sum_{i=1}^{n} |a_i-b_i|\)最小. 分析 神题啊不会. 具体证明看黄源河论文&l ...
- 【bzoj1367】[Baltic2004]sequence 可并堆
题目描述 输入 输出 一个整数R 样例输入 7 9 4 8 20 14 15 18 样例输出 13 题解 可并堆,黄源河<左偏树的特点及其应用>Page 13例题原题 #include & ...
- 【bzoj1367】[Baltic2004]sequence
2016-05-31 17:31:26 题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1367 题解:http://www.cnblogs.co ...
随机推荐
- Shell、Xterm、Gnome-Terminal、Konsole简介(转)
什么是Shell? 简单的说, Shell就是一个小程序,这个小程序可以接受来自键盘的命令并把这些命令发送到操作系统,再有系统来执行.在过去,在安装有Unix的计算机上,这是唯一的可用的交互式操作.而 ...
- 进入Material Design时代
------------------------------------------------------------------------------ GitHub:lightSky 微博: ...
- POJ 2373
原本一道挺简单的DP题,思路有了,运用单调队列,但在写单调队列时写挫了... 这道题只需要计算偶数位置的即可,这是显而易见的,我有注意过这情况,写的时候却没在意...--! 加入队列的元素应该当前no ...
- Android之——自己主动挂断电话的实现
转载请注明出处:http://blog.csdn.net/l1028386804/article/details/47072451 通过<Android之--AIDL小结>与<And ...
- 【VC编程技巧】窗口☞3.6以渐变效果加载对话框
平时我们常常能够看到非常多应用程序启动过程非常酷.什么百叶窗.渐变,各种效果,今天我们看一下怎样在程序中添加这样的效果. 一.演示样例展示: watermark/2/text/aHR0cDovL2Js ...
- iOS项目开发实战——使用Xcode6设计自己定义控件与图形
在iOS开发中,有很多控件都是Xcode默认提供的.使用这些控件是很方便的.可是因为某些须要.须要自己设计控件,那么应该怎么做呢?在Xcode6中提供了这种接口,同意开发人员高速开发自己定义控件,而且 ...
- Lua 与C/C++ 交互系列:注冊枚举enum到Lua Code中
在Lua Code中注冊C/C++的枚举很easy,就像注冊全局变量一样.我们使用枚举名称作为命名空间,来避免注冊的枚举发生冲突.注冊的枚举存储在全局环境(线程环境)中. 当在Lua Code中訪问枚 ...
- 【HAOI 2008】 移动玩具
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1054 [算法] 广度优先搜索 [代码] #include<bits/stdc+ ...
- Java-Maven:POM百科
ylbtech-Java-Maven:POM百科 Maven是以项目为中心的设计.POM(project object model)是Maven对一个单一项目的描述.没有POM的话,Maven是毫无用 ...
- [源码管理] Microsoft Visual SourceSafe 2005 下载与配置
一.VSS2005的下载地址是:百度搜索关键字:vss, 二.配置Microsoft Visual SourceSafe 2005的Internet访问 VSS2005发布以后,早就听说可以支持Int ...