征途这是一道十分经典的斜率优化

我们可以从题目中的方差来想,也就很容易的到这个式子

\[ans=m^2*\frac{\sum_{i=1}^{m}{(x_i-{\overline{x}})^2}}{m}
\]

化简就会得到

\[ans=m*\sum_{i=1}^{m}{(x_i-{\overline{x}})^2}
\]

在化简得

\[ans={m*\sum_{i=1}^{m}{x_i}^2}+{\sum_{i=1}^{m}{x_i}}
\]

经过观察,我们可以很容易发现\(\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转移要优,我们就可以得到这么一个式子

\[dp[i-1][x]+(sum[j]-sum[x])^2<dp[i-1][y]+(sum[j]-sum[y])^2
\]

化简得

\[\frac{(dp[i-1][x]+sum[x]^2)-(dp[i-1][y]+sum[y]^2)}{sum[x]-sum[y]}<2*sum[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征途 斜率优化的更多相关文章

  1. [SDOI2015][bzoj4518] 征途 [斜率优化dp]

    题面 传送门 思路 把$vm^2$展开化一下式子,可以得到这样的等价公式: $vm^2=m\sum_{i=1}^m a_i^2-\sum_{i=1}^m a_i$ 那么我们要最小化的就是$\sum_{ ...

  2. P4072 [SDOI2016](BZOJ4518) 征途 [斜率优化DP]

    题目描述 Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天晚上Pine都必须在休息站过夜.所以,一段路 ...

  3. 洛谷 P4072 [SDOI2016]征途 斜率优化DP

    洛谷 P4072 [SDOI2016]征途 斜率优化DP 题目描述 \(Pine\) 开始了从 \(S\) 地到 \(T\) 地的征途. 从\(S\)地到\(T\)地的路可以划分成 \(n\) 段,相 ...

  4. 【BZOJ4518】[Sdoi2016]征途 斜率优化

    [BZOJ4518][Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除 ...

  5. bzoj-4518 4518: [Sdoi2016]征途(斜率优化dp)

    题目链接: 4518: [Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地 ...

  6. bzoj4518[Sdoi2016]征途 斜率优化dp

    4518: [Sdoi2016]征途 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1657  Solved: 915[Submit][Status] ...

  7. 【bzoj4518】[Sdoi2016]征途 斜率优化dp

    原文地址:http://www.cnblogs.com/GXZlegend/p/6812435.html 题目描述 Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界 ...

  8. [bzoj4518][Sdoi2016]征途-斜率优化

    Brief Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地.除第m天外,每一天晚上Pine都必须 ...

  9. BZOJ 4518: [Sdoi2016]征途 [斜率优化DP]

    4518: [Sdoi2016]征途 题意:\(n\le 3000\)个数分成m组,一组的和为一个数,求最小方差\(*m^2\) DP方程随便写\(f[i][j]=min\{f[k][j-1]+(s[ ...

随机推荐

  1. bzoj4534: 基础排序算法练习题

    传送门     策爷的论文题啊……题解在这儿 我只想知道为什么这题的弱化版会出现在我们今天的%你赛里…… 题意:给你一堆操作$(l,r)$,表示将区间$(l,r)$按升序排序.以及$q$个询问,每次询 ...

  2. 例题 5-1 STL

    Raju and Meena love to play with Marbles. They have got a lot of marbles with numbers written on the ...

  3. ASP.NET MVC5 之 分部页

    1.分部页 _PartialPage.cshtml @model List<string> <a>完美世界</a> @foreach (var item in Mo ...

  4. [Qt Creator 快速入门] 第8章 界面外观

    一个完善的应用程序不仅应该有实用的功能,还要有一个漂亮的外观,这样才能使应用程序更加友好,更加吸引用户.作为一个跨平台的UI开发框架,Qt提供了强大而灵活的界面外观设计机制.这一章将学习在Qt中设计应 ...

  5. 贪心/思维题 UVA 11292 The Dragon of Loowater

    题目传送门 /* 题意:n个头,m个士兵,问能否砍掉n个头 贪心/思维题:两个数组升序排序,用最弱的士兵砍掉当前的头 */ #include <cstdio> #include <c ...

  6. MySQL客户端导入数据库脚本,字段值出现乱码解决方法

    解决方法1:在MySql安装目录下找到my.ini,将[mysql]下的default-character-set=latin1改为default-character-set=utf8,保存,然后重启 ...

  7. [ NOI 2001 ] 方程的解数

    \(\\\) \(Description\) 已知一个 \(N\) 元高次方程: \[ k_1x_1^{p_1}+k_2x_2^{p_2}+...+k_nx_n^{p_n}=0 \] 要求所有的 \( ...

  8. [ CQOI 2014 ] 数三角形

    \(\\\) Description 求 \(N\times M\) 的网格图上有多少个格点构成的三角形. 当三点共线的时候我们不认为这是一个三角形. \(n,m\le 10^4\) \(\\\) S ...

  9. javascript异步下载 Promise实现

    一般下载都是直接打开一个链接就行.var URL = 'XXXX';window.open(URL)其实这样会有些问题:1. 浏览器禁止打开新窗口,导致无法下载 那么怎么解决呢?这样: <a h ...

  10. Android O 通知栏的"running in the background"

    Android O新增的一个特性,系统会在通知栏显示当前在后台运行的应用,其实际是显示启动了前台服务的应用,并且当前应用的Activity不在前台.具体我们看下源码是怎么实现的. 1 APP调用sta ...