题目链接

题目要求使得下面这个式子最小(\(\mu=\frac{\sum_{i=1}^ma_i}{m}\)是平均数,\(a_i\)为第\(i\)段的和): $$\frac{\sum_{i-1}^m(\mu -a_i)2}{m}*m2$$

\(m\)可以乘进去,得: $$m\times\sum_{i=1}m(a_i-\frac{sum}{m})2$$

注意到其中有\(m\)个\(sum\),于是连同乘上的\(m\)一起提出来就是\(m\times\sum_{i=1}^m(a_i^2-2*a_i*\frac{sum}{m})+sum^2\)

\(\sum_{i=1}^m2*a_i\)又是一个\(sum\),于是最终化简为$$m\times\sum_{i=1}ma_i2-sum^2$$

使\(\sum_{i=1}^ma_i^2\)最小,转移方程是\(f[i][j]=f[i-1][k]+(s_j-s_k)^2\)

容易想到有单调性,用斜率优化即可。

\[f_j-f_k+s_j^2-s_k^2\leq 2*s_i*(s_j-s_k)
\]

注意初始化。。即到某点时分一组的值。

下附错误代码(化简不好导致。。)

//868kb	272ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=3005; int n,m,sum[N],f[N][2],q[N]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline int X(int j,int k){
return sum[j]-sum[k];
}
inline int Y(int j,int k,int s){
return f[j][s]-f[k][s]+(sum[j]*sum[j]-sum[k]*sum[k]);
}
inline int Calc_F(int i,int j,int s){
return f[j][s]+(sum[i]-sum[j])*(sum[i]-sum[j]);
} int main()
{
n=read(),m=read();
for(int i=1; i<=n; ++i) sum[i]=sum[i-1]+read(),f[i][0]=sum[i]*sum[i];
int p=0;
for(int j=2; j<=m; ++j)
{
int h=1,t=1; q[1]=j-1;//这可以初始化为这个,i从j枚举。
for(int i=j; i<=n; ++i)
{
while(h<t && Y(q[h+1],q[h],p)<=2*sum[i]*X(q[h+1],q[h])) ++h;
f[i][p^1]=Calc_F(i,q[h],p);
while(h<t && 1ll*Y(i,q[t],p)*X(q[t],q[t-1])<=1ll*Y(q[t],q[t-1],p)*X(i,q[t])) --t;
q[++t]=i;
}
p^=1;
}
printf("%d",m*f[n][p]-sum[n]*sum[n]); return 0;
}

错误做法:

题目要求使得下面这个式子最小(\(\mu=\frac{\sum_{i=1}^ma_i}{m}\)是平均数,\(a_i\)为第\(i\)段的和): $$\frac{\sum_{i-1}^m(\mu -a_i)2}{m}*m2$$

\(m\)可以乘进去,得: $$m\times\sum_{i=1}m(\frac{sum}{m}-a_i)2$$

那是个平方,于是写成俩,把\(m\)乘进一个去,得: $$\frac{sum2}{m}-2*a_i*sum+a_i2*m$$

其中 \(a_i=sum_{now}-sum_i\);\(\frac{sum^2}{m}\)是一个常数,直接用。

显然(...)具有决策单调性。然后展开一堆式子,得到。。不写了。

但是不对,暴力也可能和答案差1,因为\(sum^2\)不一定整除\(m\),但那个常数的位置还没法弄掉分母,所以计算\(f[]\)的时候就gg。

怕不是还能用double水

#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=3005; int n,m,w,sum[N],f[N],f2[N][N],q[N]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline int X(int j,int k){
return m*(sum[j]-sum[k]);
}
inline long long Y(int j,int k){
return f[j]-f[k]+2*(sum[j]-sum[k])*sum[n]+(sum[j]*sum[j]-sum[k]*sum[k])*m;
}
inline int Calc_F(int i,int j){
int ai=sum[i]-sum[j];
return f[j]+w-2*ai*sum[n]+ai*ai*m;
} int main()
{
n=read(),m=read();
for(int i=1; i<=n; ++i) sum[i]=sum[i-1]+read();
w=sum[n]*sum[n]/m;
memset(f2,0x3f,sizeof f2);
f2[0][0]=1;
for(int i=1; i<=m; ++i)
{
// int h=1,t=1; q[1]=0;
for(int j=i; j<=n; ++j)
{
// while(h<t && Y(q[h+1],q[h])<=2*sum[j]*X(q[h+1],q[h])) ++h;
// f[j]=Calc_F(j,q[h]);
// while(h<t && Y(i,q[t])*X(q[t],q[t-1])<=Y(q[t],q[t-1])*X(i,q[t])) --t;
// q[++t]=j;
for(int k=0; k<j; ++k)
{
int ai=sum[j]-sum[k];
f2[i][j]=std::min(f2[i][j],f2[i-1][k]+w-2*ai*sum[n]+ai*ai*m);
}
}
}
printf("%d",f2[m][n]); return 0;
}

BZOJ.4072.[SDOI2016]征途(DP 斜率优化)的更多相关文章

  1. 洛谷4072 SDOI2016征途 (斜率优化+dp)

    首先根据题目中给的要求,推一下方差的柿子. \[v\times m^2 = m\times \sum x^2 - 2 \times sum \times sum +sum*sum \] 所以\(ans ...

  2. BZOJ4518: [Sdoi2016]征途(dp+斜率优化)

    Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1875  Solved: 1045[Submit][Status][Discuss] Descript ...

  3. BZOJ_4518_[Sdoi2016]征途_斜率优化

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

  4. 【BZOJ-4518】征途 DP + 斜率优化

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

  5. BZOJ4518 Sdoi2016 征途 【斜率优化DP】 *

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

  6. 2018.09.08 bzoj4518: [Sdoi2016]征途(斜率优化dp)

    传送门 把式子展开后发现就是要求: m∗(∑i=1msum′[i])−sum[n]2" role="presentation" style="position: ...

  7. 【洛谷 P4072】 [SDOI2016]征途(斜率优化)

    好久没写斜率优化板子都忘了, 硬是交了十几遍.. 推一下柿子就能得到答案为 \[m*\sum x^2-(\sum x)^2\] 后面是个定值,前面简单dp,斜率优化一下就行了. \(f[i][j]=f ...

  8. 洛谷P4072 [SDOI2016]征途(斜率优化)

    传送门 推式子(快哭了……)$$s^2*m^2=\sum _{i=1}^m (x_i-\bar{x})^2$$ $$s^2*m^2=m*\sum _{i=1}^m x_i^2-2*sum_n\sum ...

  9. BZOJ.1492.[NOI2007]货币兑换(DP 斜率优化 CDQ分治/Splay)

    BZOJ 洛谷 如果某天能够赚钱,那么一定会在这天把手上的金券全卖掉.同样如果某天要买,一定会把所有钱花光. 那么令\(f_i\)表示到第\(i\)天所拥有的最多钱数(此时手上没有任何金券),可以选择 ...

随机推荐

  1. Python基础数据类型-字典(dict)

    Python基础数据类型-字典(dict) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 本篇博客使用的是Python3.6版本,以及以后分享的每一篇都是Python3.x版本的哟 ...

  2. fastJson顺序遍历JSON字段(转)

    fastJson在把json格式的字符串转换成JSONObject的时候,使用的是HashMap,所以排序规则是根据HASH值排序的,如果想要按照字符串顺序遍历JSON属性,需要在转换的时候指定使用L ...

  3. 阿里云Tengine和Openresty/1.11.2.3 数据对比

    HLS播放延迟测试:阿里云48s ,openresy 31s Cache-Control: max-age=300 NGINX下配置CACHE-CONTROL   Content-Length:637 ...

  4. Unable to Distribute in Xcode5?

    Blog Unable to Distribute in Xcode5? I have the  question, if this screenshot is what you getting. ( ...

  5. checkbox判断不为空

    checkbox不为空 html页面: @foreach($seals as $v) <input type="checkbox" name="seal_id[]& ...

  6. Zookeeper笔记之四字命令

    Zookeeper支持一些命令用来获取服务的状态和相关信息,因为这些命令都是四个字母的,所以一般称为四字命令. 四字命令可以使用telnet或者nc向服务器提交,使用下面这个脚本可以当做是一个简易的客 ...

  7. Burp-Suit之Interder

    登陆页面:http://localhost/pentest/brute/login.php 设置代理,使用Burp截断: 发送到Intruder进行爆破,这里我先说明一下Intruder页面 Inte ...

  8. 针对Jigsaw勒索软件的解锁工具

    针对Jigsaw勒索软件的解锁工具 据了解, 用户的计算机系统一旦感染了勒索软件Jigsaw,如果用户没有在一个小时之内支付赎金(0.4个比特币,价值约为150美金),那么恶意软件将会把系统中的上千份 ...

  9. VAE(Variational Autoencoder)的原理

    Kingma, Diederik P., and Max Welling. "Auto-encoding variational bayes." arXiv preprint ar ...

  10. 善用backtrace解决大问题【转】

    转自:https://www.2cto.com/kf/201107/97270.html 一.用途: 主要用于程序异常退出时寻找错误原因 二.功能: 回溯堆栈,简单的说就是可以列出当前函数调用关系 三 ...