[原题描述以及提交地址]:http://acm.tongji.edu.cn/problem?pid=10011

[题目大意]

  给定两个长度为N的序列,要给这两个序列的数连线。连线只能在两个序列之间进行,且连线不能交叉,每个数最多只能选一次。连线从左到右进行,每次连线收益为这两个数的乘积。对于两个序列,都有:每段连续的没被选中的数的和的平方为损失。

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

防剧透

[解题思路]

  O(n^4):

  f[i][j]代表a序列前i个数,b序列前j个数中i,j必选所得到的最优收益。

  f[i][j] = a[i] * b[j] + max(f[k][l] - (suma[i - 1] - suma[k])^2 - (sumb[j - 1] - sumb[l])^2) {0 < k < i, 0 < l < j}

===================================================================================

  O(n^3):

  可以发现对于k + 1...i 以及 l + 1...j 这两段数之间可以再连线,而且答案不会更劣。

  于是有k == i - 1 or l == j - 1

  f[i][j] = a[i] * b[j] + max(f[k][j - 1] - (suma[i - 1] - suma[k])^2,f[i - 1][l] - (sumb[j - 1] - sumb[l])^2) {0 < k < i, 0 < l < j}

===================================================================================

  O(n^2):

  事实上以上的方程是可以用斜率优化的。只不过是同时依赖于两个斜率优化方程而已。于是,对于每个i,j开一个单调队列,维护即可。

===================================================================================

Postscript:打斜率优化的时候一定要注意等号,而且最好从凸包的角度来理解,来实现,比较不容易出错。

#include <cstdio>
#include <algorithm>
#include <deque>
const int N = 1000 + 9;
typedef long long ll;
int n,a[N],b[N],i,j,t;
ll suma[N],sumb[N],f[N][N];
std::deque<int> qi[N],qj[N];
inline ll sqr(const ll x){return x*x;}
inline ll calci(const int x)
{return f[i - 1][x] - sqr(sumb[j - 1] - sumb[x]);}
inline ll calcj(const int x)
{return f[x][j - 1] - sqr(suma[i - 1] - suma[x]);}
inline ll Xi(const int k,const int l)
{return f[i - 1][k] - sqr(sumb[k]) - (f[i - 1][l] - sqr(sumb[l]));}
inline ll Yi(const int k,const int l)
{return sumb[l] - sumb[k];}
inline ll Xj(const int k,const int l)
{return f[k][j - 1] - sqr(suma[k]) - (f[l][j - 1] - sqr(suma[l]));}
inline ll Yj(const int k,const int l)
{return suma[l] - suma[k];}
int main()
{
#ifndef ONLINE_JUDGE
freopen("sxbk.in","r",stdin);
freopen("sxbk.out","w",stdout);
#endif
scanf("%d",&n);
for (i = 1; i <= n; ++i) {
scanf("%d",a+i);
suma[i] = suma[i - 1] + a[i];
}
for (i = 1; i <= n; ++i) {
scanf("%d",b+i);
sumb[i] = sumb[i - 1] + b[i];
}
for (i = 1; i <= n; ++i) {
for (j = 1; j <= n; ++j) {
while (qi[i - 1].size() > 1 && calci(qi[i - 1].front()) <= calci(qi[i - 1][1])) qi[i - 1].pop_front();
while (qj[j - 1].size() > 1 && calcj(qj[j - 1].front()) <= calcj(qj[j - 1][1])) qj[j - 1].pop_front();
f[i][j] = - sqr(suma[i - 1]) - sqr(sumb[j - 1]);
if ((i - 1) && qi[i - 1].size()) f[i][j] = std::max(f[i][j],calci(qi[i - 1].front()));
if ((j - 1) && qj[j - 1].size()) f[i][j] = std::max(f[i][j],calcj(qj[j - 1].front()));
f[i][j] += a[i] * b[j];
while ((t = qi[i - 1].size()) > 1 && Xi(qi[i - 1][t - 2],qi[i - 1].back()) * Yi(qi[i - 1].back(),j) >= Xi(qi[i - 1].back(),j) * Yi(qi[i - 1][t - 2],qi[i - 1].back())) qi[i - 1].pop_back();
while ((t = qj[j - 1].size()) > 1 && Xj(qj[j - 1][t - 2],qj[j - 1].back()) * Yj(qj[j - 1].back(),i) >= Xj(qj[j - 1].back(),i) * Yj(qj[j - 1][t - 2],qj[j - 1].back())) qj[j - 1].pop_back();
if (i - 1) qi[i - 1].push_back(j);
if (j - 1) qj[j - 1].push_back(i);
}
}
ll ans = -0x7fffffff;
for (int i = 1; i <= n; ++i)
ans = std::max(ans,std::max(f[i][n] - sqr(suma[n] - suma[i]),f[n][i] - sqr(sumb[n] - sumb[i])));
printf("%I64d\n",ans);
}

  

[USACO Special 2007 Chinese Competition]The Bovine Accordion and Banjo Orchestra的更多相关文章

  1. 【BZOJ1713】[Usaco2007 China]The Bovine Accordion and Banjo Orchestra 音乐会 斜率优化

    [BZOJ1713][Usaco2007 China]The Bovine Accordion and Banjo Orchestra 音乐会 Description Input 第1行输入N,之后N ...

  2. BZOJ_1713_[Usaco2007 China]The Bovine Accordion and Banjo Orchestra 音乐会_斜率优化

    BZOJ_1713_[Usaco2007 China]The Bovine Accordion and Banjo Orchestra 音乐会_斜率优化 Description Input 第1行输入 ...

  3. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  4. [Elite 2008 Dec USACO]Jigsaw Puzzles

    #include <iostream> #include <cstdio> #include <cstring> using namespace std; #def ...

  5. Delphi QC 记录

    各网友提交的 QC: 官方网址 说明 备注 https://quality.embarcadero.com/browse/RSP-12985 iOS device cannot use indy id ...

  6. TOJ1693(Silver Cow Party)

    Silver Cow Party   Time Limit(Common/Java):2000MS/20000MS     Memory Limit:65536KByte Total Submit: ...

  7. 一些基于jQuery开发的控件

    基于jQuery开发,非常简单的水平方向折叠控件.主页:http://letmehaveblog.blogspot.com/2007/10/haccordion-simple-horizontal-a ...

  8. TOJ 1690 Cow Sorting (置换群)

    Description Farmer John's N (1 ≤ N ≤ 10,000) cows are lined up to be milked in the evening. Each cow ...

  9. TOJ1698: Balanced Lineup

    Description For the daily milking, Farmer John's N cows (1 ≤ N ≤ 50,000) always line up in the same ...

随机推荐

  1. axios超时重发

    axios的超时是在response中处理的,所以要在response中添加拦截器: axios.interceptors.response.use(undefined, function axios ...

  2. npm错误总结

    You cannot publish over the previously published version 1.0.1." : xxx 发布时一定要修改package.json的版本号 ...

  3. 十个迅速提升JQuery性能的技巧

    本文提供即刻提升你的脚本性能的十个步骤.不用担心,这并不是什么高深的技巧.人人皆可运用!这些技巧包括: 使用最新版本 合并.最小化脚本 用for替代each 用ID替代class选择器 给选择器指定前 ...

  4. Vim使用小记(二)插件管理

    By francis_hao    Mar 8,2017 Vundle Vundle,全称为Vim bundle,是一个插件管理器.可以对vim插件进行安装和卸载. Vundle的安装方法看这里[参考 ...

  5. 近期对于windows服务的理解

    1.APP.config的作用   在开发环境下时,根目录下的APP.config里面会填写一些参数之类的.当生成之后,这些参数将会被自动生成在*.exe文件目录中.如图: 其中,.exe文件为Win ...

  6. ng的ngModel用来处理表单操作

    https://segmentfault.com/a/1190000009126012

  7. jwplayer 部署方案1

    <body> <div id="my_player" data_src="http://xx.com/jwplayer/uploads/test.mp4 ...

  8. 五分钟搞懂Vuex

    这段时间一直在用vue写项目,vuex在项目中也会依葫芦画瓢使用,但是总有一种朦朦胧胧的感觉.于是决定彻底搞懂它. 看了一下午的官方文档,以及资料,才发现vuex so easy! 作为一个圈子中的人 ...

  9. mybatis 关系映射

    一:订单商品数据模型 1.数据库执行脚本 创建数据库表代码: 1 CREATE TABLE items ( 2 id INT NOT NULL AUTO_INCREMENT, 3 itemsname ...

  10. 如何实现用户id生成一个唯一邀请码

    #如何实现用户id生成一个唯一邀请码 #创建验证码 function createCode($user_id) { static $source_string = 'E5FCDG3HQA4B1NOPI ...