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()产生子进程, ...