题解-[CEOI2017]Building Bridges
有 \(n\) 个桥墩,高 \(h_i\) 重 \(w_i\)。连接 \(i\) 和 \(j\) 消耗代价 \((h_i-h_j)^2\),用不到的桥墩被拆除,代价为 \(w_i\)。求使 \(1\) 与 \(n\) 联通的最小代价。
数据范围:\(2\le n\le 10^5\),\(0\le h_i,|w_i|\le 10^6\)。
非常经典的李超线段树维护 \(\texttt{dp}\) 的题目,小蒟蒻来分享一下。
很明显 \(w_i\) 是大片大片消耗的,所以记 \(s_i=\sum_{j=1}^i w_j\)。
令 \(f_i\) 表示连接到第 \(i\) 个桥墩的最小代价。可以野蛮推式:
f_i=&\min\{f_j+(h_i-h_j)^2+s_{i-1}-s_j\}\\
=&\min\{f_j+h_i^2-2h_ih_j+h_j^2+s_{i-1}-s_j\}\\
=&h_i^2+s_{i-1}+\min\{f_j-2h_ih_j+h_j^2-s_j\}\\
\end{split}\\
\]
这貌似是个斜率优化式子,但蒟蒻不管,用李超线段树怎么做呢?
考虑李超线段树的作用:多条线段(直线),求单点最值。
发现这个 \(j\) 有很多,而 \(i\) 就只有当前一个:所以可以 \(i\) 对应单点,\(j\) 对应线。换句话说,可以把每个 \(f_i\) 求出来后添加一条直线。
\textrm{let } x=h_i \textrm{ to get }\min\{f_j-2h_ih_j+h_j^2-s_j\}\textrm{.}\\
\]
这题有几个坑,本来是应该由你来快乐地调试的,但是既然写了题解,蒟蒻就放出来了:
- 因为要计算 \(h_j^2\),所以要开 \(\texttt{long long}\) 或用 \(1ll\) 乘之。
- 这个李超线段树是权值线段树,下标要开 \(10^6\) 个,节点个数要开 \(4\cdot 10^6\) 个。
这个做法貌似有点辜负了这题的难度,但是蒟蒻只会这么做。蒟蒻讲不清楚,还是放个蒻蒻的代码吧:
#include <bits/stdc++.h>
using namespace std;
//Start
#define lng long long
#define db double
#define mk make_pair
#define pb push_back
#define fi first
#define se second
#define rz resize
const int inf=0x3f3f3f3f;
const lng INF=0x3f3f3f3f3f3f3f3f;
//Data
const int N=1e5,M=1e6;
int n,h[N+7];
lng w[N+7],f[N+7];
//Lichaotree
typedef pair<lng,lng> line;
lng g(line&li,int x){return li.fi*x+li.se;}
int inter(line&la,line&lb){return db(lb.se-la.se)/(la.fi-lb.fi);}
line v[(M<<2)+7];
void add(line li,int k=1,int l=0,int r=M){
int mid((l+r)>>1);
lng ly1=g(li,l),ry1=g(li,r),ly=g(v[k],l),ry=g(v[k],r);
if(ly1>=ly&&ry1>=ry);
else if(ly1<=ly&&ry1<=ry) v[k]=li;
else {
int in=inter(li,v[k]);
if(ly1<=ly){
if(in<=mid) add(li,k<<1,l,mid);
else add(v[k],k<<1|1,mid+1,r),v[k]=li;
} else {
if(in>mid) add(li,k<<1|1,mid+1,r);
else add(v[k],k<<1,l,mid),v[k]=li;
}
}
}
lng get(int x,int k=1,int l=0,int r=M){
lng res(g(v[k],x));
if(l==r) return res;
int mid((l+r)>>1);
if(mid>=x) res=min(res,get(x,k<<1,l,mid));
else res=min(res,get(x,k<<1|1,mid+1,r));
return res;
}
//Main
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",&h[i]);
for(int i=1;i<=n;i++) scanf("%lld",&w[i]),w[i]+=w[i-1];
fill(v+1,v+(M<<2)+1,mk(0,INF));
f[1]=0,add(mk(-2ll*h[1],1ll*h[1]*h[1]-w[1]));
for(int i=2;i<=n;i++){
f[i]=1ll*h[i]*h[i]+w[i-1]+get(h[i]);
add(mk(-2ll*h[i],f[i]+1ll*h[i]*h[i]-w[i]));
}
printf("%lld\n",f[n]);
return 0;
}
祝大家学习愉快!
题解-[CEOI2017]Building Bridges的更多相关文章
- Luogu4655 [CEOI2017]Building Bridges
Luogu4655 [CEOI2017]Building Bridges 有 \(n\) 根柱子依次排列,每根柱子都有一个高度.第 \(i\) 根柱子的高度为 \(h_i\) . 现在想要建造若干座桥 ...
- ceoi2017 Building Bridges(build)
Building Bridges(build) 题目描述 A wide river has nn pillars of possibly different heights standing out ...
- 洛谷.4655.[CEOI2017]Building Bridges(DP 斜率优化 CDQ分治)
LOJ 洛谷 \(f_i=s_{i-1}+h_i^2+\min\{f_j-s_j+h_j^2-2h_i2h_j\}\),显然可以斜率优化. \(f_i-s_{i-1}-h_i^2+2h_ih_j=f_ ...
- [CEOI2017]Building Bridges
题目 斜率优化思博题,不想写了 之后就一直\(95\)了,于是靠肮脏的打表 就是更新了一下凸壳上二分斜率的写法,非常清爽好写 就当是挂个板子了 #include<algorithm> #i ...
- loj#2483. 「CEOI2017」Building Bridges 斜率优化 cdq分治
loj#2483. 「CEOI2017」Building Bridges 链接 https://loj.ac/problem/2483 思路 \[f[i]=f[j]+(h[i]-h[j])^2+(su ...
- HDU 4584 Building bridges (水题)
Building bridges Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) ...
- LOJ 2483: 洛谷 P4655: 「CEOI2017」Building Bridges
题目传送门:LOJ #2483. 题意简述: 有 \(n\) 个数,每个数有高度 \(h_i\) 和价格 \(w_i\) 两个属性. 你可以花费 \(w_i\) 的代价移除第 \(i\) 个数(不能移 ...
- loj#2483. 「CEOI2017」Building Bridges(dp cdq 凸包)
题意 题目链接 Sol \[f[i], f[j] + (h[i] - h[j])^2 + (w[i - 1] - w[j]))\] 然后直接套路斜率优化,发现\(k, x\)都不单调 写个cdq就过了 ...
- @loj - 2483@「CEOI2017」Building Bridges
目录 @desription@ @solution@ @accepted code@ @details@ @another solution@ @another code@ @desription@ ...
随机推荐
- opencv 学习1 gui属性
1.图像 先定一个小目标学习两个个功能 1.学习怎样读入一副画像,以及如何保存一副图像 2.学习三个函数cv2.imread(), cv2.imshow() , cv2.imwrite() 1.1读入 ...
- mysql之分区表
1.分区表概述: 1.分区表的主要意义在于,对于表结构进行划分,不同的数据进入不同的分区中,以便于在查询过程中,只查找指定分区的数据,减少数据库扫描的数据量. 2.虽然从逻辑上看分区表是一张表,但是底 ...
- rados put striper功能的调试
前言 之前对于striper这个地方的功能并没研究太多,只是知道这个里面可以以条带方式并行的去写对象,从而加大并发性来提高性能,而默认的条带数目为1,也就是以对象大小去写,并没有条带,所以不是很好感觉 ...
- 基于docker部署ceph以及修改docker image
前言 容器和ceph的结合已经在一些生产环境当中做了尝试,容器的好处就是对运行环境的一个封装,传统的方式是集成为ISO,这个需要一定的维护量,而容器的相关操作会简单很多,也就有了一些尝试,个人觉得如果 ...
- Jar 和 war 区别
jar包:对于学习java的人来说应该并不陌生.我们也经常使用也一些jar包.其实jar包就是java的类进行编译生成的class文件就行打包的压缩包而已.里面就是一些class文件.当我们自己使用m ...
- 每天一个linux命令之top
每天一个linux命令之top 转 https://www.linuxprobe.com/chapter-02.html 在图2-6中,top命令执行结果的前5行为系统整体的统计信息,其所代表的含义 ...
- MySQL数据库 | MySQL调优|MySQL底层原理|MySQL零基础新手教程
MySQL数据库安装 一.Windows 环境下安装 A.下载 MySQL Select Operating System: Microsoft Windows 快捷下载:mysql-8.0.22-w ...
- 企业级工作流解决方案(十四)--集成Abp和ng-alain--自动化脚本
对于.net方向,做过自动化的,应该没有人不熟悉msbuild吧,非常强大的代码编译工具,.net平台的编译工作都是交给他来完成的,包括.net core的命令,本质上都是调用msbuild来执行的 ...
- Android light系统分析
光线系统包括:背光,闪光,led指示灯 一.内核层 Led-class.c (kernel-3.10\drivers\leds) 这个文件给HAL层提供接口 led_brightn ...
- macbook上安装虚拟机软件如何操作?
很多用户都不太熟悉苹果系统,用惯了Windows之后再过渡到MacOS难免会有些不习惯.为了使我们又可以用回那些熟悉的Windows应用,比较常见的办法就是安装macbook虚拟机.下面小编就教大家一 ...