题目大意

给定一个长度为n的正整数序列,令修改一个数的代价为修改前后两个数的绝对值之差,求用最小代价将序列转换为不减序列。

其中,n满足小于500000,序列中的正整数小于10^9

题解(引自mzx神犇的题解)

本次test跪0了,尴尬

解法1(40分)

考虑dp

设到第i个数为止,序列中数全部<=j的最小代价为f[i][j]

可以推出f[i][j]=min{f[i-1][j]+|ai-j|,f[i][j-1]}

解法2(60分)

是对于第一个dp思路的优化

既然数字是固定的,可以离散化,降低空间时间复杂度

解法3(100分)

斜率优化dp,用线段树维护斜率

解法4(100分,引自zyz大神的解法)

玄学的单调栈问题

由于区间中的数是单调不减的,所以最后得出的序列一定是一段一段的各个数值相等的序列区间

而要最小化代价,可以取这些区间中的中位数作为标准来修改每个区间,并用单调栈求解最优的区间划分

还需要用动态开点线段树维护区间中位数

orz

#include <stdio.h>
#include <string.h> const int MAXN = 5e5 + ;
const int INF = 1e9 + ; inline int abs(int x){
return x < ? -x : x;
} int Rin(){
int x = , c = getchar(), f = ;
for(; c < || c > ; c = getchar())
if(!(c ^ ))
f = -;
for(; c > && c < ; c = getchar())
x = (x << ) + (x << ) + c - ;
return x * f;
} namespace Arcueid{
struct Node{
Node *l, *r;
int s;
Node(){}
Node(Node *_l, Node *_r, int _s) : l(_l), r(_r), s(_s) {}
}*_nil = new Node(), *nil = (*_nil = Node(_nil, _nil, ), _nil), pool[MAXN * ], *top = pool;
Node *reborn(){
*top = Node(nil, nil, );
return top++;
}
void death(Node *&pr, Node *pl, int l, int r, int v){
pr = reborn(); *pr = *pl; pr->s++;
if(l + < r){
int mid = (l + r) >> ;
v < mid ? death(pr->l, pl->l, l, mid, v) : death(pr->r, pl->r, mid, r, v);
}
}
int secret(Node *pr, Node *pl, int l, int r, int k){
if(l + == r)return l;
int c = pr->l->s - pl->l->s, mid = (l + r) >> ;
return c < k ? secret(pr->r, pl->r, mid, r, k - c) : secret(pr->l, pl->l, l, mid, k);
}
int feel(Node *pr, Node *pl){
int c = pr->s - pl->s;
return secret(pr, pl, , INF, (c + ) >> );
}
} Arcueid::Node *rt[MAXN] = {Arcueid::nil}; namespace moon{
void cause(){
int n, a[MAXN], stay = , night[MAXN], shiki[MAXN]; // freopen("chen.in", "r", stdin);
// freopen("chen.out", "w", stdout); n = Rin();
for(int i = ; i <= n; i++){
a[i] = Rin();
Arcueid::death(rt[i], rt[i-], , INF, a[i]);
for(; stay > && Arcueid::feel(rt[i], rt[night[stay - ]]) < shiki[stay - ]; stay--);
shiki[stay] = Arcueid::feel(rt[i], rt[night[stay-]]);
night[stay++] = i;
}
long long ans = ;
for(int i = ; i < stay; i++)
for(int j = night[i - ] + ; j <= night[i]; j++)
ans += abs(a[j] - shiki[i]); printf("%lld\n", ans);
// fclose(stdin);
// fclose(stdout);
}
} int main(){
moon::cause();
return ;
}

解法5(100分)

其实这个解法更玄

但最简洁的往往是最好的

#include <queue>
#include <vector>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> using namespace std; inline int read(){
int x = , c = getchar(), f = ;
for(; c < || c > ; c=getchar())
if(!(c ^ ))
f = -;
for(; c > && c < ; c=getchar())
x = (x << ) + (x << ) + c - ;
return x * f;
} long long ans = ;
priority_queue<int> q; int main(){
int n = read();
for(int i = ; i <= n; i++){
int x = read();
q.push(x);
ans += q.top() - x;
if(x ^ q.top())
q.pop();
}
printf("%lld\n", ans);
return ;
}

[2016湖南长沙培训Day4][前鬼后鬼的守护 chen] (动态开点线段树+中位数 or 动规 or 贪心+堆优化)的更多相关文章

  1. BZOJ5059 前鬼后鬼的守护 【堆扩展】*

    BZOJ5059 前鬼后鬼的守护 Description 八云紫的式神八云蓝有一张符卡名为[式神-前鬼后鬼的守护],这张符卡的弹幕为BOSS从两侧向自机发射大玉,大玉后面跟着一些小玉,形成一个&quo ...

  2. BZOJ 5059: 前鬼后鬼的守护 可并堆 左偏树 数学

    https://www.lydsy.com/JudgeOnline/problem.php?id=5059 题意:将原序列{ai}改为一个递增序列{ai1}并且使得abs(ai-ai1)的和最小. 如 ...

  3. BZOJ 5059 前鬼后鬼的守护

    题解: 解法一:用函数斜率什么的,不会,留坑 解法二: 某一个序列都变成一个值那么中位数最优 加入一个元素,与前面那一段区间的中位数比较 x>=mid什么事也不做 x<mid合并两端区间 ...

  4. 探究C语言中的前++和后++

    小波带您探究c语言中的前++与后++: 欢迎吐槽,欢迎加QQ463431476. 欢迎关注!  现在来探究: 咱们先看第一个 i被赋值0,i++(后++)并没有输出1.   现在i被赋值0,++i,也 ...

  5. HMM 自学教程(七)前向后向算法

    本系列文章摘自 52nlp(我爱自然语言处理: http://www.52nlp.cn/),原文链接在 HMM 学习最佳范例,这是针对 国外网站上一个 HMM 教程 的翻译,作者功底很深,翻译得很精彩 ...

  6. STL——前闭后开区间表示法和function call 操作符

    前开后闭开区间表示法[) 任何一个STL算法,都需要获得由一对迭代器(泛型指针)所标示的区间,用以表示操作范围,这一对迭代器所标示的是个所谓的前闭后开区间,以[first,last)表示,也就是说,整 ...

  7. HMM 前向后向算法(转)

    最近研究NLP颇感兴趣,但由于比较懒,所以只好找来网上别人的比较好的博客,备份一下,也方便自己以后方便查找(其实,一般是不会再回过头来看的,嘿嘿 -_-!!) 代码自己重新写了一遍,所以就不把原文代码 ...

  8. 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率

    隐马尔科夫模型HMM(一)HMM模型 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率 隐马尔科夫模型HMM(三)鲍姆-韦尔奇算法求解HMM参数(TODO) 隐马尔科夫模型HMM(四)维特比算法 ...

  9. jquery取前、后、父、子元素

    前.prev(); 后.next(); 父.parent(); 子.children(); 注意:前的前是.prev().prev(),例如前元素无i,但前的前的i元素有i,不能写成.prev('i' ...

随机推荐

  1. Java中,方法的重写、重载的区别,以及多态的实例

    首先我们要明白什么是重写和重载 重写(override):子类方法覆盖了父类的方法.    (类与类之间继承的关系) 例:父类代码 public class Deng { public void Qi ...

  2. 第一弹:Java 中创建对象的4种方式

    Java 是面向对象的语言,不可避免的,"对象"这个概念是 Java 语言的核心部分,这里来简单讨论一下在 Java 中创建一般对象的方法. 总结下来有以下4种创建对象的方法: 使 ...

  3. 【夯实PHP系列】购物车代码说明PHP的匿名函数

    1. 定义:匿名函数(Anonymous functions),也叫闭包函数(closures),允许 临时创建一个没有指定名称的函数.最经常用作回调函数(callback)参数的值.当然,也有其它应 ...

  4. 火狐浏览器如何js关闭窗口的几种解决方法

    今天在项目上有一个页面要求在几秒后自动关闭,想着还比较简单,用window.close()就可以了,但是用IE/谷歌/火狐浏览器试了一下,发现IE可以,谷歌用网上的兼容方法也可以实现,但是火狐这里卡住 ...

  5. IOS开发基础知识--碎片15

    1:将自定义对象转化成NsData存入数据库 要转为nsdata自定义对象要遵循<NSCoding>的协议,然后实现encodeWithCoder,initwithcode对属性转化,实例 ...

  6. n个元素的入栈顺序有多少种出栈顺序?

    问题:w1.w2.w3.w4.w5,5个元素将会按顺序入栈,求出栈顺序有多少种情况. 先写一下结论方便记忆: 1个元素:1种 2个元素:2种 3个元素:5种 4个元素:14种 5个元素:42种 简单的 ...

  7. CoreLocation定位技术

    CoreLocation框架可用于定位设备当前经纬度,通过该框架,应用程序可通过附近的蜂窝基站,WIFI信号或者GPS等信息计算用户位置.      iOS定位支持的3种模式.      (1)GPS ...

  8. miterLimit和lineJoin属性

    <!DOCTYPE HTML> <head> <meta charset = "utf-8"> <title>starGirl< ...

  9. INITIAL参数设置导致TRUNCATE TABLE不能降低高水位线案例

    在一个数据库使用下面SQL找出了一批需要降低高水位线的表,其中有几个表没有数据,于是我打算用TRUNCATE来降低高水位线HWM SELECT a.owner,        a.segment_na ...

  10. Linux下开启关闭SeLinux

    SELinux (Security-Enhanced Linux) in Fedora is an implementation of mandatory access control in the ...