bzoj4518征途 斜率优化
征途这是一道十分经典的斜率优化
我们可以从题目中的方差来想,也就很容易的到这个式子
\]
化简就会得到
\]
在化简得
\]
经过观察,我们可以很容易发现\(\sum_{i=1}^{m}{x_i}\)是定值,同时\(m*\sum_{i=1}^{m}{x_i}^2\)中的\(m\)也是定值,所以我们的目标就是让\(\sum_{i=1}^{m}{{x_i}^2}\)最小
我们令\(dp[i][j]=\min\{dp[i-1][j-k]+(sum[j]-sum[j-k])^2\}\)
其中我们令\(sum[i]=\sum_{k=1}^{i}{val_k}\)
其中\(val_k\)是第k段路的长度
由此我们就可得到一个开o2可以卡成80的代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cctype>
using namespace std;
long long n,m,dp[3001][3001],sum[3001];
const int BufferSize=100*1000;
char buffer[BufferSize],*head,*tail;
bool not_EOF=true;
inline char Getchar(){
if(not_EOF and head==tail){
int len=fread(buffer,1,BufferSize,stdin);
not_EOF=len!=0;
head=buffer,tail=head+len;
}
return not_EOF?*head++:-1;
}
inline long long rd(){
long long x=0,s=1;
char c=Getchar();
for(;!isdigit(c) and not_EOF;c=Getchar()) if(c=='-') s=-1;
for(; isdigit(c) and not_EOF;c=Getchar()) x=(x<<1)+(x<<3)+(c^48);
return s*x;
}
inline void scan(char *str){
char c=Getchar();
for(; isspace(c) and not_EOF;c=Getchar());
for(;!isspace(c) and not_EOF;c=Getchar()) *(str++)=c;
*str=0;
}
int main(){
memset(dp,0x7f,sizeof(dp));
n=rd(),m=rd();
for(int i=1;i<=n;i++){
long long x=rd();
sum[i]=sum[i-1]+x;
}
dp[0][0]=-sum[n]*sum[n];
for(int i=0;i<m;i++){
for(int j=0;j<=n;j++){
for(int k=0;k<=n-j;k++){
dp[i+1][j+k]=min(dp[i+1][j+k],dp[i][j]+(sum[j+k]-sum[j])*(sum[j+k]-sum[j])*m);
}
}
}
printf("%lld",dp[m][n]);
return 0;
}
但是我们可以发现这个代码是\(O(nm)\)的,显然我是不可以接受的。其实我们可以很容易证明这个的决策单调性,于是我们就可以用斜率优化
我们令\(y<x<i\)
因为决策单调性,所以对于决策\(i\)时,k转移必定比j转移要优,我们就可以得到这么一个式子
\]
化简得
\]
这样我们就可以用斜率又花了
我们把\(p_1(dp[i-1][x]+sum[x]^2,sum[x])\)和\(p_2(dp[i-1][y]+sum[y]^2,sum[y])\)看做\(p_1,p_2\)两个点,刚刚那个式子也就是它们所连线的斜率表达式
我们也就可以得到这样的代码
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
long long n,m,dp[3001][3001],sum[3001];
int q[3001];
double getpoint(int i,int p){return (double)dp[i][p]+sum[i]*sum[i];}
double slope(int j,int k,int p){return (double)(getpoint(j,p)-getpoint(k,p))/(double)(sum[j]-sum[k]);}
int main(){
scanf("%lld%lld",&n,&m);
for(int i=1;i<=n;i++){
long long x;
scanf("%lld",&x);
sum[i]=sum[i-1]+x;
dp[i][1]=sum[i]*sum[i];
}
for(int k=2;k<=m;k++){
int head=1,tail=0;
for(int i=1;i<=n;i++){
while(head<tail and slope(q[head],q[head+1],k-1)<2*sum[i])head++;
int j=q[head];
dp[i][k]=dp[j][k-1]+(sum[j]-sum[i])*(sum[j]-sum[i]);
while(head<tail and slope(q[tail],q[tail-1],k-1)>slope(q[tail],i,k-1))tail--;
q[++tail]=i;
}
}
printf("%lld",m*dp[n][m]-sum[n]*sum[n]);
return 0;
}
bzoj4518征途 斜率优化的更多相关文章
- [SDOI2015][bzoj4518] 征途 [斜率优化dp]
题面 传送门 思路 把$vm^2$展开化一下式子,可以得到这样的等价公式: $vm^2=m\sum_{i=1}^m a_i^2-\sum_{i=1}^m a_i$ 那么我们要最小化的就是$\sum_{ ...
- P4072 [SDOI2016](BZOJ4518) 征途 [斜率优化DP]
题目描述 Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天晚上Pine都必须在休息站过夜.所以,一段路 ...
- 洛谷 P4072 [SDOI2016]征途 斜率优化DP
洛谷 P4072 [SDOI2016]征途 斜率优化DP 题目描述 \(Pine\) 开始了从 \(S\) 地到 \(T\) 地的征途. 从\(S\)地到\(T\)地的路可以划分成 \(n\) 段,相 ...
- 【BZOJ4518】[Sdoi2016]征途 斜率优化
[BZOJ4518][Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除 ...
- bzoj-4518 4518: [Sdoi2016]征途(斜率优化dp)
题目链接: 4518: [Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地 ...
- bzoj4518[Sdoi2016]征途 斜率优化dp
4518: [Sdoi2016]征途 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1657 Solved: 915[Submit][Status] ...
- 【bzoj4518】[Sdoi2016]征途 斜率优化dp
原文地址:http://www.cnblogs.com/GXZlegend/p/6812435.html 题目描述 Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界 ...
- [bzoj4518][Sdoi2016]征途-斜率优化
Brief Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天晚上Pine都必须 ...
- BZOJ 4518: [Sdoi2016]征途 [斜率优化DP]
4518: [Sdoi2016]征途 题意:\(n\le 3000\)个数分成m组,一组的和为一个数,求最小方差\(*m^2\) DP方程随便写\(f[i][j]=min\{f[k][j-1]+(s[ ...
随机推荐
- bzoj4534: 基础排序算法练习题
传送门 策爷的论文题啊……题解在这儿 我只想知道为什么这题的弱化版会出现在我们今天的%你赛里…… 题意:给你一堆操作$(l,r)$,表示将区间$(l,r)$按升序排序.以及$q$个询问,每次询 ...
- 例题 5-1 STL
Raju and Meena love to play with Marbles. They have got a lot of marbles with numbers written on the ...
- ASP.NET MVC5 之 分部页
1.分部页 _PartialPage.cshtml @model List<string> <a>完美世界</a> @foreach (var item in Mo ...
- [Qt Creator 快速入门] 第8章 界面外观
一个完善的应用程序不仅应该有实用的功能,还要有一个漂亮的外观,这样才能使应用程序更加友好,更加吸引用户.作为一个跨平台的UI开发框架,Qt提供了强大而灵活的界面外观设计机制.这一章将学习在Qt中设计应 ...
- 贪心/思维题 UVA 11292 The Dragon of Loowater
题目传送门 /* 题意:n个头,m个士兵,问能否砍掉n个头 贪心/思维题:两个数组升序排序,用最弱的士兵砍掉当前的头 */ #include <cstdio> #include <c ...
- MySQL客户端导入数据库脚本,字段值出现乱码解决方法
解决方法1:在MySql安装目录下找到my.ini,将[mysql]下的default-character-set=latin1改为default-character-set=utf8,保存,然后重启 ...
- [ NOI 2001 ] 方程的解数
\(\\\) \(Description\) 已知一个 \(N\) 元高次方程: \[ k_1x_1^{p_1}+k_2x_2^{p_2}+...+k_nx_n^{p_n}=0 \] 要求所有的 \( ...
- [ CQOI 2014 ] 数三角形
\(\\\) Description 求 \(N\times M\) 的网格图上有多少个格点构成的三角形. 当三点共线的时候我们不认为这是一个三角形. \(n,m\le 10^4\) \(\\\) S ...
- javascript异步下载 Promise实现
一般下载都是直接打开一个链接就行.var URL = 'XXXX';window.open(URL)其实这样会有些问题:1. 浏览器禁止打开新窗口,导致无法下载 那么怎么解决呢?这样: <a h ...
- Android O 通知栏的"running in the background"
Android O新增的一个特性,系统会在通知栏显示当前在后台运行的应用,其实际是显示启动了前台服务的应用,并且当前应用的Activity不在前台.具体我们看下源码是怎么实现的. 1 APP调用sta ...