Description

给定一个序列\(t_1,t_2,\cdots,t_n\),求一个递增序列\(z_1<z_2<...<z_n\),

使得 \(R=|t_1−z_1|+|t_2−z_2|+\cdots +|t_n−z_n|\) 的值最小。求\(R\)

Analysis

1.转化

\(z_1<z_2<...<z_n\)的小于号不爽

转化成\(z_1\le z_2\le...\le z_n\)

我们令\(i<j\), 根据条件我们有\(z_j-z_i\ge j-i\)

移一下项则\(z_i-i \le z_j-j\)

我们令\(x_i=t_i-i\),\(y_i=z_i-i\)

用\(x,y\)相减是等价的, 且转化成了\(y_1\le y_2\le...\le y_n\)

后面我们用\(x,y\)代替\(t,z\)

2.尝试简单化的题目

假如x单调递增,那么\(x_i=y_i\)

假如x单调递减呢,\(y_1=y_2=\cdots=y_n=\)x中位数

注:单调递增可以表示为多个单调递减

证明:

假如有条件\(y_1=y_2=\cdots =y_n\), 这个证明不难

现在稍微加一步

①设\(y_i\)变小,则\(y_1...y_{i-1}\)都变小

若\(i<mid\),R显然变大

若\(i>mid\),R变大的点数比变小的点数要多

②设\(y_i\)变大,同理

Solution

对于每个点i一开始属于块i,块中答案\(ans_i=x_i\)

从前往后扫,维护单调队列, 出现y变小的时候退栈

将两个区间合并,合并后区间的ans变为两块一起的中位数

一直合并, 知道上一个区间的y比当前区间的y小

证明:

首先,合并过的区间一定含有至少一个长度大于1的单调减区间

且一开始,合并过的区间里每个区间都只有一个单调减区间

现在我们要证明的就是两个(多个)单调减区间拼在一起的最优答案也是中位数

跟前面的证明是类似的,如图(绿线辅助线,黑线表示两个单调减区间)

(考虑移动ans那条线)

正确性

归纳, 初始答案为\(y_1=x_1\)

考虑当前加入\(x_i\)

如果\(y_{i-1}\le x_i\), 显然, 直接令\(y_i = x_i\)是最优的

当出现\(y_{i-1}\gt x_i\)时, 通过合并操作可以使得答案变优, \(y\)最大值变小

Code

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <cctype>
#include <algorithm>
using namespace std;
typedef long long LL;
const int M=1000007;
inline int rd(){
int x=0;bool f=1;char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=1;
for(;isdigit(c);c=getchar()) x=x*10+c-48;
return f?x:-x;
} int n;
int x[M];
struct node{
int l,r;
int rt;
node(int ll=0,int rr=0,int __=0){
l=ll;r=rr;
rt=__;
}
}que[M];
int tt; int val[M];
int dist[M];
int sz[M];
int lc[M],rc[M]; int merge(int x,int y){
if(!x) return y;
if(!y) return x;
if(val[x]<val[y]) swap(x,y);
rc[x]=merge(rc[x],y);
if(dist[rc[x]]>dist[lc[x]]) swap(lc[x],rc[x]);
dist[x]=dist[rc[x]]+1;
sz[x]=sz[lc[x]]+sz[rc[x]]+1;
return x;//*****
} void pop(int &x){
x=merge(lc[x],rc[x]);
} int main(){
int i,j;
n=rd();
for(i=1;i<=n;i++) x[i]=rd()-i;
for(i=1;i<=n;i++){
que[++tt]=node(i,i,i);
val[i]=x[i];
dist[i]=sz[i]=1;
while(tt>1&&val[que[tt].rt]<val[que[tt-1].rt]){
tt--;
que[tt].r=que[tt+1].r;
que[tt].rt=merge(que[tt].rt,que[tt+1].rt);
while(sz[que[tt].rt]*2>(que[tt].r-que[tt].l+2)){
pop(que[tt].rt);
}
}
}
LL ans=0;
for(i=1;i<=n;i++)
for(j=que[i].l;j<=que[i].r;j++)
ans+=abs(val[que[i].rt]-x[j]);
printf("%lld\n",ans);
return 0;
}

bzoj 1367 - sequence的更多相关文章

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

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

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

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

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

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

  4. bzoj 1367: [Baltic2004]sequence

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

  5. bzoj 1367 [ Baltic 2004 ] sequence —— 左偏树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1367 好题啊!论文上的题: 论文上只给出了不下降序列的求法: 先考虑特殊情况,如果原序列上升 ...

  6. BZOJ 1367 [Baltic2004]sequence (可并堆)

    题面:BZOJ传送门 题目大意:给你一个序列$a$,让你构造一个递增序列$b$,使得$\sum |a_{i}-b_{i}|$最小,$a_{i},b_{i}$均为整数 神仙题.. 我们先考虑b不递减的情 ...

  7. sequence(bzoj 1367)

    Description Input Output 一个整数R Sample Input 794820141518 Sample Output 13 HINT 所求的Z序列为6,7,8,13,14,15 ...

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

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

  9. 【BZOJ 1049】 1049: [HAOI2006]数字序列 (LIS+动态规划)

    1049: [HAOI2006]数字序列 Description 现在我们有一个长度为n的整数序列A.但是它太不好看了,于是我们希望把它变成一个单调严格上升的序列.但是不希望改变过多的数,也不希望改变 ...

随机推荐

  1. JS控制台的使用

    1.快捷键F12可直接进入控制台(或者单机右键->检查)用于bug调试 2.控制台如图: Elements:表示所有的js元素 Console:常用的有如下几个功能: ①console.log: ...

  2. EasyUI获取正在编辑状态行的索引

    function getRowIndex(target){ var tr = $(target).closest("tr.datagrid-row"); return paseIn ...

  3. ES6箭头函数基本用法

    ES6箭头函数基本用法 ``` window.onload = function(){ alert(abc); } //箭头函数 window.onload = ()=>{ alert(&quo ...

  4. Python中列表的深浅拷贝

    copy_lst = [ ('py对象三要素',), ('== 比较运算符',), ('is 身份运算符',), ('小数据池',), ('列表的浅拷贝',), ('列表的深拷贝',), ] py对象 ...

  5. Network of Schools POJ - 1236 (强联通)

    一些学校连接到了一个计算机网络.网络中的学校间有如下约定:每个学校维护一个列表,当该学校收到软件或信息后将会转发给列表中的所有学校(也就是接收方列表).需要注意的是如果B学校在A学校的接收方列表中,A ...

  6. Three Steps to Migrate Group Policy Between Active Directory Domains or Forests Using PowerShell

    Three Steps Ahead Have you ever wished that you had three legs? Imagine how much faster you could ru ...

  7. 在 Amazon AWS 搭建及部署网站:(三)开发及部署环境

    服务器已经搭建好,网站也开始运行了.那么如何方便地部署代码呢? 最基本的方式,就是使用 SFTP 向网站目录直接部署.这种方法的缺点是版本控制不便,在上传时也无法方便的比较代码变化. 用SVN来部署是 ...

  8. [网站公告]又拍云API故障造成图片无法上传(已恢复)

    大家好,18:00左右开始,又拍云API出现故障,调用图片上传API时出现错误:“The remote server returned an error: (403) Forbidden.”,造成图片 ...

  9. 37、iamgeview 图层叠加

    1 Drawable d1 = new BitmapDrawable(circleBitmap); Drawable d2 = login.this.getResources().getDrawabl ...

  10. 【Jump Game】cpp

    题目: Given an array of non-negative integers, you are initially positioned at the first index of the ...