[2016湖南长沙培训Day4][前鬼后鬼的守护 chen] (动态开点线段树+中位数 or 动规 or 贪心+堆优化)
题目大意
给定一个长度为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 贪心+堆优化)的更多相关文章
- BZOJ5059 前鬼后鬼的守护 【堆扩展】*
BZOJ5059 前鬼后鬼的守护 Description 八云紫的式神八云蓝有一张符卡名为[式神-前鬼后鬼的守护],这张符卡的弹幕为BOSS从两侧向自机发射大玉,大玉后面跟着一些小玉,形成一个&quo ...
- BZOJ 5059: 前鬼后鬼的守护 可并堆 左偏树 数学
https://www.lydsy.com/JudgeOnline/problem.php?id=5059 题意:将原序列{ai}改为一个递增序列{ai1}并且使得abs(ai-ai1)的和最小. 如 ...
- BZOJ 5059 前鬼后鬼的守护
题解: 解法一:用函数斜率什么的,不会,留坑 解法二: 某一个序列都变成一个值那么中位数最优 加入一个元素,与前面那一段区间的中位数比较 x>=mid什么事也不做 x<mid合并两端区间 ...
- 探究C语言中的前++和后++
小波带您探究c语言中的前++与后++: 欢迎吐槽,欢迎加QQ463431476. 欢迎关注! 现在来探究: 咱们先看第一个 i被赋值0,i++(后++)并没有输出1. 现在i被赋值0,++i,也 ...
- HMM 自学教程(七)前向后向算法
本系列文章摘自 52nlp(我爱自然语言处理: http://www.52nlp.cn/),原文链接在 HMM 学习最佳范例,这是针对 国外网站上一个 HMM 教程 的翻译,作者功底很深,翻译得很精彩 ...
- STL——前闭后开区间表示法和function call 操作符
前开后闭开区间表示法[) 任何一个STL算法,都需要获得由一对迭代器(泛型指针)所标示的区间,用以表示操作范围,这一对迭代器所标示的是个所谓的前闭后开区间,以[first,last)表示,也就是说,整 ...
- HMM 前向后向算法(转)
最近研究NLP颇感兴趣,但由于比较懒,所以只好找来网上别人的比较好的博客,备份一下,也方便自己以后方便查找(其实,一般是不会再回过头来看的,嘿嘿 -_-!!) 代码自己重新写了一遍,所以就不把原文代码 ...
- 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率
隐马尔科夫模型HMM(一)HMM模型 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率 隐马尔科夫模型HMM(三)鲍姆-韦尔奇算法求解HMM参数(TODO) 隐马尔科夫模型HMM(四)维特比算法 ...
- jquery取前、后、父、子元素
前.prev(); 后.next(); 父.parent(); 子.children(); 注意:前的前是.prev().prev(),例如前元素无i,但前的前的i元素有i,不能写成.prev('i' ...
随机推荐
- 基于小脚丫DDS 调频 调幅 调相 切换波形 AD5601输出模拟波形
先讲讲里面的矩阵键盘,矩阵键盘列有下拉电阻,默认全为0000,默认行输入为1111,当有按键按下的时候,列输入会被拉高,这时控制行的输出做行扫描,电子琴用key_flag_r0电平作为使能,这里用ke ...
- easyui datagrid 动态操作editor 的方法
easyui本身是不提供这么细节的功能的,需要我们自己拓展下: 在easyui.min.js中扩展: $.extend($.fn.datagrid.methods, { addEditor : fun ...
- C#中 反射中的Assembly(装载程序集):
反射中的Assembly(装载程序集):可以通过Assembly的信息来获取程序的类,实例等编程需要用到的信息. String assemblyName = @"NamespaceRef& ...
- jQuery:年月日三级联动
<%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default3.aspx.cs ...
- [翻译]Java HashMap工作原理
大部分Java开发者都在使用Map,特别是HashMap.HashMap是一种简单但强大的方式去存储和获取数据.但有多少开发者知道HashMap内部如何工作呢?几天前,我阅读了java.util.Ha ...
- [ASP.NET MVC] ASP.NET Identity登入技术剖析
[ASP.NET MVC] ASP.NET Identity登入技术剖析 前言 ASP.NET Identity是微软所贡献的开源项目,用来提供ASP.NET的验证.授权等等机制.本篇文章介绍ASP. ...
- jQuery基础知识总结
1. jQuery基本概念介绍 1.1 什么是jQuery 一个javascript库,把常用方法写到一个js文件中,需要的时候直接调用即可 学习jQuery就是学习一些方法 ...
- php每天一题:怎么在不使用第三个变量的情况下交换两个变量的值
$a = 'php'; $b = 'my'; list($a,$b) = array($b,$a); echo $a,$b; 很简单,大家试一下是不是交换了!
- 天津政府应急系统之GIS一张图(arcgis api for flex)讲解(九)地图定位模块
config.xml文件的配置如下: <widget label="地图定位" config="widgets/esri/Location/LocationWidg ...
- ks
http://www.codeproject.com/Articles/207820/The-Repository-Pattern-with-EF-code-first-Dependen