CF932F Escape Through Leaf(DP,斜率优化)
SB 题。
写出 DP 方程:\(f_i\) 表示从 \(i\) 跳的最小值。
\(i\) 是叶子就是 \(0\),否则就是选个子树中的 \(v\),\(f_i=\min(f_v+a_ib_v)\)。
至于优化,求出每个子树中的凸包就行了。启发式合并保证复杂度。
复杂度 \(O(n\log^2 n)\)。
没错,我又用了回家路线那又臭又长的写法。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=100010;
#define FOR(i,a,b) for(int i=(a);i<=(b);i++)
#define ROF(i,a,b) for(int i=(a);i>=(b);i--)
#define MEM(x,v) memset(x,v,sizeof(x))
inline int read(){
int x=0,f=0;char ch=getchar();
while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
struct line{
int k;
ll b;
bool operator<(const line &l)const{
if(k!=l.k) return k>l.k;
return b<l.b;
}
};
struct point{
double x;
int k;
ll b;
bool operator<(const point &p)const{return x<p.x;}
};
int n,el,head[maxn],to[maxn*2],nxt[maxn*2],A[maxn],B[maxn];
ll f[maxn];
set<line> inter[maxn];
set<point> hull[maxn];
inline void add(int u,int v){to[++el]=v;nxt[el]=head[u];head[u]=el;}
double interx(line l1,line l2){
return l1.k==l2.k?1e18:1.0*(l2.b-l1.b)/(l1.k-l2.k);
}
void remove(int id,set<line>::iterator it){
set<line>::iterator it1=it,it2=it;it2++;
if(it1!=inter[id].begin()){
it1--;
hull[id].erase((point){interx(*it,*it1),it1->k,it1->b});
it1++;
}
if(it2!=inter[id].end()) hull[id].erase((point){interx(*it,*it2),it->k,it->b});
if(it1!=inter[id].begin() && it2!=inter[id].end()){
it1--;
hull[id].insert((point){interx(*it1,*it2),it1->k,it1->b});
}
inter[id].erase(it);
}
void insert(int id,line l){
set<line>::iterator it=inter[id].insert(l).first;
set<line>::iterator it1=it,it2=it;it2++;
if(it1!=inter[id].begin()){
it1--;
if(it1->k==it->k) return void(inter[id].erase(*it));
it1++;
}
if(it1!=inter[id].begin() && it2!=inter[id].end()){
it1--;
if(interx(*it,*it1)>=interx(*it,*it2)) return void(inter[id].erase(*it));
it1++;
}
if(it1!=inter[id].begin()){
it1--;
hull[id].insert((point){interx(*it,*it1),it1->k,it1->b});
it1++;
}
if(it2!=inter[id].end()) hull[id].insert((point){interx(*it,*it2),it->k,it->b});
if(it1!=inter[id].begin() && it2!=inter[id].end()){
it1--;
hull[id].erase((point){interx(*it1,*it2),it1->k,it1->b});
it1++;
}
it=it1=inter[id].find(l);
while(it1!=inter[id].begin()){
it1--;
if(it1==inter[id].begin()) break;
it2=it1;it2--;
if(interx(*it2,*it)<=interx(*it2,*it1)) remove(id,it1);
else break;
it=it1=inter[id].find(l);
}
it=it1=inter[id].find(l);it1++;
while(it1!=inter[id].end()){
it2=it1;it2++;
if(it2!=inter[id].end() && interx(*it2,*it)>=interx(*it2,*it1) || it1->k==it->k) remove(id,it1);
else break;
it=it1=inter[id].find(l);it1++;
}
}
void dfs(int u,int F){
for(int i=head[u];i;i=nxt[i]){
int v=to[i];
if(v==F) continue;
dfs(v,u);
if(inter[u].size()<inter[v].size()) inter[u].swap(inter[v]),hull[u].swap(hull[v]);
for(set<line>::iterator it=inter[v].begin();it!=inter[v].end();it++) insert(u,*it);
inter[v].clear();hull[v].clear();
}
if(!inter[u].empty()){
set<point>::iterator it=hull[u].lower_bound((point){A[u],0,0});
int k;
ll b;
if(it==hull[u].end()){
set<line>::iterator it=inter[u].end();it--;
k=it->k;b=it->b;
}
else k=it->k,b=it->b;
f[u]=1ll*k*A[u]+b;
}
insert(u,(line){B[u],f[u]});
}
int main(){
n=read();
FOR(i,1,n) A[i]=read();
FOR(i,1,n) B[i]=read();
FOR(i,1,n-1){
int u=read(),v=read();
add(u,v);add(v,u);
}
dfs(1,0);
FOR(i,1,n) printf("%lld ",f[i]);
}
CF932F Escape Through Leaf(DP,斜率优化)的更多相关文章
- CF932F Escape Through Leaf
CF932F Escape Through Leaf 首先, $ O(n^2) $ dp 是很显然的,方程长这样: \[dp[u] = min\{dp[v] + a_u\times b_v\} \] ...
- 【BZOJ-4518】征途 DP + 斜率优化
4518: [Sdoi2016]征途 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 230 Solved: 156[Submit][Status][ ...
- 【BZOJ-3437】小P的牧场 DP + 斜率优化
3437: 小P的牧场 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 705 Solved: 404[Submit][Status][Discuss ...
- 【BZOJ-1010】玩具装箱toy DP + 斜率优化
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 8432 Solved: 3338[Submit][St ...
- 【BZOJ】1096: [ZJOI2007]仓库建设(dp+斜率优化)
http://www.lydsy.com/JudgeOnline/problem.php?id=1096 首先得到dp方程(我竟然自己都每推出了QAQ)$$d[i]=min\{d[j]+cost(j+ ...
- BZOJ 1096: [ZJOI2007]仓库建设(DP+斜率优化)
[ZJOI2007]仓库建设 Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L公司一般把产品直接堆放在 ...
- 学渣乱搞系列之dp斜率优化
学渣乱搞系列之dp斜率优化 By 狂徒归来 貌似dp的斜率优化一直很难搞啊,尤其是像我这种数学很挫的学渣,压根不懂什么凸包,什么上凸下凸的,哎...说多了都是泪,跟wdd讨论了下,得出一些结论.本文很 ...
- DP斜率优化总结
目录 DP斜率优化总结 任务安排1 任务计划2 任务安排3 百日旅行 DP斜率优化总结 任务安排1 首先引入一道题,先\(O(N^2)\)做法:分别预处理出\(T_i,C_i\)前缀和\(t[i],c ...
- HDU 3507 [Print Article]DP斜率优化
题目大意 给定一个长度为\(n(n \leqslant 500000)\)的数列,将其分割为连续的若干份,使得 $ \sum ((\sum_{i=j}^kC_i) +M) $ 最小.其中\(C_i\) ...
- dp斜率优化
算法-dp斜率优化 前置知识: 凸包 斜率优化很玄学,凭空讲怎么也讲不好,所以放例题. [APIO2014]序列分割 [APIO2014]序列分割 给你一个长度为 \(n\) 的序列 \(a_1,a_ ...
随机推荐
- Tensorflow faster rcnn系列一
注意:本文主要是学习用,发现了一个在faster rcnn训练流程写的比较详细的博客. 大部分内容来自以下博客连接:https://blog.csdn.net/weixin_37203756/arti ...
- vuex 源码分析(七) module和namespaced 详解
当项目非常大时,如果所有的状态都集中放到一个对象中,store 对象就有可能变得相当臃肿. 为了解决这个问题,Vuex允许我们将 store 分割成模块(module).每个模块拥有自己的 state ...
- shell脚本语言与linux命令的联系与区别
使用linux肯定是要会使用命令的,就算提供有用户界面,绝大部分功能还是要通过命令行去操作的.而shell脚本语言也是运行在linux上的脚本语言,对于服务器运维人员也是几乎必须要掌握的.而shell ...
- SimdJsonSharp:每秒解析千兆字节的JSON
SimdJsonSharp: Parsing gigabytes of JSON per second C# version of lemire/simdjson (by Daniel Lemire ...
- Window权限维持(七):安全支持提供者
安全支持提供程序(SSP)是Windows API,用于扩展Windows身份验证机制.LSASS进程正在Windows启动期间加载安全支持提供程序DLL.这种行为使红队的攻击者可以删除一个任意的SS ...
- mongodb实现文件存储系统
前言:这种坑很深呀,要对应mongodb的版本跟php支持的版本,然后,如果要用composer安装第三方的库,一定要一一对应的 正片开始! 开发环境: 系统:window 开发语言:php+apac ...
- deepin可视化程序打不开问题排查方法
anyconnect是一个VPN软件,在deepin系统下安装完成之后,并不能够直接使用,点击启动图标之后没有反应. 要想分析问题,必须从命令行入手,错误会打印在控制台. 如何根据一个图标来找到一个程 ...
- GitBash上传项目出现[fatal: remote origin already exists.]问题解决方案
问题截图如下: 当出现这个问题时,不要慌,只需要输入 git remote rm origin 就可以解决,输入完前面的命令后再次 输入 git remote add origin https://. ...
- Mysql EF Core 快速构建 Web Api
(1)首先创建一个.net core web api web项目; (2)因为我们使用的是ef连接mysql数据库,通过NuGet安装MySql.Data.EntityFrameworkCore,以来 ...
- system execl
1. system(); int system(const char *command); 在Windows下,用来调用常用的Dos命令 在Linux下,system()会调用fork()产生子进程, ...