【题解】征途 SDOI 2016 BZOJ 4518
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4518
首先推式子,我们用$x_i$表示第$i$段的路程,$sum$表示总路程,根据方差和平均数的定义,有:
$ \Large sum = \sum\limits_{i=1}^{m}{x_i} \\ $
$ \Large \bar{x} = \frac{sum}{m} \\ $
$ \Large ans = \frac{ \sum\limits_{i=1}^{m}{ (x_i - \bar{x})^2 } }{m} \cdot m^2 \\ $
然后我们把式子展开化简一番,就有了:
$ \Large ans = ( \sum\limits_{i=1}^{m}{ (x_i - \frac{sum}{m})^2 } ) \cdot m \\ $
$ \Large ans = ( \sum\limits_{i=1}^{m}{ \frac{ (m \cdot x_i - sum)^2 }{m^2} } ) \cdot m \\ $
$ \Large ans = \sum\limits_{i=1}^{m}{ \frac{ m^2 \cdot x_i^2 - 2 \cdot m \cdot x_i \cdot sum + sum^2 }{m} } \\ $
$ \Large ans = sum^2 + \sum\limits_{i=1}^{m}{ (m \cdot x_i^2 - 2 \cdot x_i \cdot sum) } \\ $
于是我们的目标就是最小化这个式子,看上去很像个dp?没错。
先抛开前面的常数$sum^2$,我们下面的dp过程中计算的是后面那个求和式子的最小值。
设$f(i,j)$表示我们把前$j$个数划分成了$i$段的最小值,$dist(i,j)$表示从第$i$个数到第$j$个数的和,用$d$数组存放前缀和,于是有:
$ \Large dist(i,j) = d_j - d_{i-1} \\ $
$ \Large f(i,j) = f(i-1,k) + m \cdot dist(k+1,j)^2 - 2 \cdot dist(k+1,j) \cdot sum \\ $
对这个式子展开并化简,得到:
$ \Large f(i,j) = f(i-1,k) + m \cdot (d_j - d_k)^2 - 2 \cdot (d_j - d_k) \cdot sum \\ $
$ \Large f(i,j) = f(i-1,k) + m \cdot (d_j^2 - 2 \cdot d_j \cdot d_k + d_k^2 ) - 2 \cdot d_j \cdot sum + 2 \cdot d_k \cdot sum \\ $
再整理一下式子,我们就可以进行斜率优化dp了,下面的$x$和$y$表示状态对应的点的横纵坐标,$k$表示状态对应的斜率,$b$表示状态对应的常数项。
$ \Large b = m \cdot d_j^2 - 2 \cdot d_j \cdot sum \\ $
$ \Large y = f(i-1,k) + m \cdot d_k^2 + 2 \cdot d_k \cdot sum \\ $
$ \Large x = d_k \\ $
$ \Large k = 2 \cdot m \cdot d_j \\ $
$ \Large f(i,j) = y - x \cdot k + b \\ $
我们的任务是最小化$f(i,j)$,且$x,y,k$是递增的,于是就用单调队列维护一个下凸壳就好啦。
代码:
#include <cstring>
#include <cstdio>
#include <algorithm> using namespace std;
typedef long long ll;
const double EPS = 1e-;
const int MAXN = ; struct Point {
ll x, y;
Point( ll x = , ll y = ):x(x),y(y){}
Point operator-( const Point &rhs ) const {
return Point(x-rhs.x, y-rhs.y);
}
};
typedef Point Vector;
double Cross( Vector v, Vector w ) {
return (double)v.x*w.y - (double)v.y*w.x;
} namespace MonoQ { // 维护下凸壳的单调队列
Point P[MAXN];
int head, tail;
void clear() {
head = tail = ;
}
void insert( Point Q ) { // 插入一个点
while( tail-head >= && Cross(P[tail-]-P[tail-], Q-P[tail-]) < EPS ) --tail;
P[tail++] = Q;
}
Point query( ll k ) { // 根据斜率查询最小值
while( tail-head >= && P[head].y - P[head].x*k > P[head+].y - P[head+].x*k ) ++head;
return P[head];
}
} ll n, m, d[MAXN] = {}; ll f[][MAXN];
ll getx( int i ) { // 这四个函数用于获取一个状态对应的几何信息
return d[i];
}
ll gety( int cur, int i ) {
return f[cur][i] + m*d[i]*d[i] + *d[i]*d[n];
}
ll getb( int i ) {
return m*d[i]*d[i] - *d[i]*d[n];
}
ll getk( int i ) {
return *m*d[i];
}
void solve() {
int cur = ;
memset( f, 0x3f, sizeof(f) );
f[cur][] = ;
for( int i = ; i <= m; ++i ) {
cur ^= ;
MonoQ::clear();
MonoQ::insert( Point(getx(i-), gety(cur^, i-)) );
for( int j = i; j <= n; ++j ) {
Point P = MonoQ::query(getk(j));
f[cur][j] = P.y - P.x*getk(j) + getb(j);
MonoQ::insert( Point(getx(j), gety(cur^, j)) );
}
}
printf( "%lld\n", f[cur][n] + d[n]*d[n] ); // 别忘了最后加上sum^2
} int main() {
scanf( "%lld%lld", &n, &m );
for( int i = ; i <= n; ++i ) {
scanf( "%lld", d+i );
d[i] += d[i-]; // 前缀和
}
solve();
return ;
}
【题解】征途 SDOI 2016 BZOJ 4518的更多相关文章
- 【BZOJ 4518】【SDOI 2016 Round1 Day2 T3】征途
比较明显的斜率优化DP,省选时因为时间太紧张和斜率DP写得不熟等原因只写了60分的暴力DP,其实当时完全可以对拍来检验标算的正确,但是我当时too naive- 很快打完了,调了将近一晚上QAQ,因为 ...
- 征途 bzoj 4518
征途(1s 256MB)journey [问题描述] Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天 ...
- 【BZOJ 4598】【SDOI 2016 Round2 Day1 T3】模式字符串
2016-05-21因为BZOJ上“ 数据文件太过巨大,仅提供前三组数据测试.”所以我考场上写的60分的点分治交上去也A了. 我的这个点分治的时间复杂度是$O(Tnmlogn)$的,听题解时没听懂$O ...
- 【BZOJ 4517】【SDOI 2016 Round1 Day2 T2】排列计数
本蒟蒻第一次没看题解A的题竟然是省选$Round1$ $Day2$ $T2$ 这道组合数学题. 考试时一开始以为是莫队,后来想到自己不会组合数的一些公式,便弃疗了去做第三题,,, 做完第三题后再回来看 ...
- 动态规划(决策单调优化):BZOJ 4518 [Sdoi2016]征途
4518: [Sdoi2016]征途 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 532 Solved: 337[Submit][Status][ ...
- BZOJ 4518 [Sdoi2016]征途(分治DP)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=4518 [题目大意] 给出一个数列,分成m段,求方差最小,答案乘上m的平方. [题解] ...
- BZOJ 4518: [Sdoi2016]征途 [斜率优化DP]
4518: [Sdoi2016]征途 题意:\(n\le 3000\)个数分成m组,一组的和为一个数,求最小方差\(*m^2\) DP方程随便写\(f[i][j]=min\{f[k][j-1]+(s[ ...
- ●BZOJ 4518 [Sdoi2016]征途
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4518 题解: 斜率优化DP 首先看看最后答案的形式: 设a[i]为第i天走的距离,那么 $A ...
- bzoj 4518: [Sdoi2016]征途
Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天晚上Pine都必须在休息站过夜 ...
随机推荐
- 做程序开发的你如果经常用Redis,这些问题肯定会遇到
分布式缓存Redis是一种支持Key-Value等多种数据结构的存储系统.可用于缓存.事件发布或订阅.高速队列等多种场景.Redis使用ANSI C语言编写,提供字符串(String).哈希(Hash ...
- JAVA学习笔记--字符串概述
一.String类 String类代表字符串,是由字符构成的一个序列.创建String对象的方法很简单,有以下几种: 1)用new来创建: String s1 = new String("m ...
- Python基础灬dict&set
字典dict 字典使用键-值(key-value)存储,具有极快的查找速度. dict基本操作 取值 a_dict = {'name': 'jack', 'age': 18} print(a_dict ...
- python基础知识-7-内存、深浅、文件操作
python其他知识目录 1.一些对内存深入理解的案例 以下列举列表,列表/字典/集合这些可变类型都是一样的原理 变量是个地址,指向存储数据的内存空间的地址,它的实质就相当于c语言里的指针.变量和数据 ...
- JSON.stringify处理对象时的问题
1. JSON.stringify({entry_key: 'test', entry_detail: undefined}) 结果 为 "{"entry_key": & ...
- git实验
四.实例应用 应用1.现有项目移植到git代管 进入目标项目,进行git初始化: 初始化:git init 修改config:git config -- local user.name '名称' 和 ...
- ES6的新特性(2)——let 与 const 增强变量声明
let 与 const 增强变量声明 ES6 新增了let命令,用来声明局部变量.它的用法类似于var,但是所声明的变量,只在let命令所在的代码块内有效,而且有暂时性死区的约束. 先看个var的常见 ...
- 冲刺ing-6
第六次Scrum冲刺 队员完成的任务 队员 完成任务 吴伟华 Leangoo的看板截图,燃尽图 蔺皓雯 编写博客,界面设计 蔡晨旸 界面设计 曾茜 测试 鲁婧楠 学习后端设计 杨池宇 界面设计 成员遇 ...
- tomcat开发环境配置
1.环境配置教程 环境变量.安装版.配置版 2.编写启动tomcat的批处理文件 3.改变端口 4.虚拟目录
- POJ 2063 Investment 滚动数组+完全背包
题目链接: http://poj.org/problem?id=2063 题意: 你现在有现金m元,你要做n年的存款投资,给你k种投资方式,每种需要现金vi元,能获得xi元的理论,一年到期后你要利用拿 ...