【CF932F】Escape Through Leaf

题意:给你一棵n个点的树,每个点有树形ai和bi,如果x是y的祖先,则你可以从x花费$a_x\times b_y$的费用走到y(费用可以为负)。对于每个点,求从这个点开始走到某个叶子节点的最小费用。

$n\le 10^5,|a_i|,|b_i|\le 10^5$

题解:用f[x]表示x的答案,显然f[x]=min{f[y]+a[x]*b[y]}是一个凸包,我们可以用set维护凸包,到时候自底向上做一次启发式合并就行了(也可以线段树合并)。

用叉积会爆long long差评~

#include <cstdio>
#include <cstring>
#include <iostream>
#include <set>
#include <algorithm>
using namespace std;
const int maxn=100010;
typedef long long ll;
struct node
{
ll x,y;
node () {}
node (ll a,ll b) {x=a,y=b;}
bool operator < (const node &a) const {return (x==a.x)?(y<a.y):(x<a.x);}
long double operator * (const node &a) const {return (long double)x*a.y-(long double)y*a.x;}
node operator - (const node &a) const {return node(x-a.x,y-a.y);}
};
set<node> s[maxn];
set<node>::iterator it,iit;
int n,cnt,tot;
int to[maxn<<1],nxt[maxn<<1],head[maxn],siz[maxn],rt[maxn];
ll va[maxn],vb[maxn],f[maxn];
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-') f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
return ret*f;
}
inline void add(int a,int b)
{
to[cnt]=b,nxt[cnt]=head[a],head[a]=cnt++;
}
inline void insert(int a,node x)
{
it=s[a].lower_bound(x);
if(it!=s[a].end()&&(*it).x==x.x) s[a].erase(it),it=s[a].lower_bound(x);
if(it!=s[a].begin())
{
it--;
if((*it).x==x.x) return ;
it++;
}
node l,r;
if(it!=s[a].end()&&it!=s[a].begin())
{
r=*it,it--,l=*it;
if((x-l)*(r-x)<=0) return ;
}
while(1)
{
it=s[a].lower_bound(x);
if(it==s[a].end()) break;
l=*it,it++;
if(it==s[a].end()) break;
r=*it;
if((l-x)*(r-l)<=0) s[a].erase(l);
else break;
}
while(1)
{
it=s[a].lower_bound(x);
if(it==s[a].begin()) break;
it--,r=*it;
if(it==s[a].begin()) break;
it--,l=*it;
if((r-l)*(x-r)<=0) s[a].erase(r);
else break;
}
s[a].insert(x);
}
inline int merge(int a,int b)
{
if(s[a].size()<s[b].size()) swap(a,b);
for(iit=s[b].begin();iit!=s[b].end();iit++) insert(a,*iit);
s[b].clear();
return a;
}
void dfs(int x,int fa)
{
rt[x]=++tot;
for(int i=head[x],y;i!=-1;i=nxt[i]) if(to[i]!=fa)
{
y=to[i],dfs(y,x),rt[x]=merge(rt[x],rt[y]);
}
if(!s[rt[x]].size()) f[x]=0;
else
{
int l=-100001,r=100001,mid;
node a,b;
while(l<r)
{
mid=(l+r)>>1;
it=s[rt[x]].lower_bound(node(mid,-1ll<<60));
if(it==s[rt[x]].begin())
{
l=mid+1;
continue;
}
if(it==s[rt[x]].end())
{
r=mid;
continue;
}
b=*it,it--,a=*it;
if((b.y-a.y)<=-va[x]*(b.x-a.x)) l=mid+1;
else r=mid;
}
it=s[rt[x]].lower_bound(node(l-1,-1ll<<60));
f[x]=va[x]*(*it).x+(*it).y;
}
insert(rt[x],node(vb[x],f[x]));
}
int main()
{
//freopen("cf932F.in","r",stdin);
//freopen("cf932F.out","w",stdout);
n=rd();
int i,a,b;
memset(head,-1,sizeof(head));
for(i=1;i<=n;i++) va[i]=rd();
for(i=1;i<=n;i++) vb[i]=rd();
for(i=1;i<n;i++) a=rd(),b=rd(),add(a,b),add(b,a);
dfs(1,0);
for(i=1;i<=n;i++) printf("%lld ",f[i]);
return 0;
}//3 1 -1 1 -1 -1 -1 1 2 2 3

【CF932F】Escape Through Leaf 启发式合并set维护凸包的更多相关文章

  1. CF932F Escape Through Leaf

    CF932F Escape Through Leaf 首先, $ O(n^2) $ dp 是很显然的,方程长这样: \[dp[u] = min\{dp[v] + a_u\times b_v\} \] ...

  2. CF932F Escape Through Leaf 斜率优化、启发式合并

    传送门 \(DP\) 设\(f_i\)表示第\(i\)个节点的答案,\(S_i\)表示\(i\)的子节点集合,那么转移方程为\(f_i = \min\limits_{j \in S_i} \{a_i ...

  3. CF932F Escape Through Leaf(DP,斜率优化)

    SB 题. 写出 DP 方程:\(f_i\) 表示从 \(i\) 跳的最小值. \(i\) 是叶子就是 \(0\),否则就是选个子树中的 \(v\),\(f_i=\min(f_v+a_ib_v)\). ...

  4. Luogu5290 十二省联考2019春节十二响(贪心+启发式合并)

    考虑链的做法,显然将两部分各自从大到小排序后逐位取max即可,最后将根计入.猜想树上做法相同,即按上述方式逐个合并子树,最后加入根.用multiset启发式合并即可维护.因为每次合并后较小集合会消失, ...

  5. loj#6041. 「雅礼集训 2017 Day7」事情的相似度(后缀自动机+启发式合并)

    题面 传送门 题解 为什么成天有人想搞些大新闻 这里写的是\(yyb\)巨巨说的启发式合并的做法(虽然\(LCT\)的做法不知道比它快到哪里去了--) 建出\(SAM\),那么两个前缀的最长公共后缀就 ...

  6. Codeforces Round #463 F. Escape Through Leaf (李超线段树合并)

    听说正解是啥 set启发式合并+维护凸包+二分 根本不会啊 , 只会 李超线段树合并 啦 ... 题意 给你一颗有 \(n\) 个点的树 , 每个节点有两个权值 \(a_i, b_i\) . 从 \( ...

  7. 【bzoj3510】首都 LCT维护子树信息(+启发式合并)

    题目描述 在X星球上有N个国家,每个国家占据着X星球的一座城市.由于国家之间是敌对关系,所以不同国家的两个城市是不会有公路相连的. X星球上战乱频发,如果A国打败了B国,那么B国将永远从这个星球消失, ...

  8. 【BZOJ3123】森林(主席树,启发式合并)

    题意:一个带点权的森林,要求维护以下操作: 1.询问路径上的点权K大值 2.两点之间连边 n,m<=80000 思路:如果树的结构不发生变化只需要维护DFS序 现在因为树的结构发生变化,要将两棵 ...

  9. BZOJ2888 资源运输(LCT启发式合并)

    这道题目太神啦! 我们考虑他的每一次合并操作,为了维护两棵树合并后树的重心,我们只好一个一个的把节点加进去.那么这样一来看上去似乎就是一次操作O(nlogn),但是我们拥有数据结构的合并利器--启发式 ...

随机推荐

  1. Java编程思想学习笔记——复用类

    前言 复用代码是Java众多引人注目的功能之一. 达到复用代码的方法有: 组合:新的类由现有类的对象所组成.(复用现有代码的功能,而非它的形式) 继承:按照现有类的类型组建新类.(不改变现有类的形式, ...

  2. org.apache.http.client.methods.HttpGet 转到定义找不到源代码

    例如 org.apache.http.client.methods.HttpGet ,提示没有源码 到这里下载 http://hc.apache.org/downloads.cgi Source 4. ...

  3. SQL语句:一个表,通过一个字段查找另外一个字段不相同值

    select * from [dbo].[Sys_MemberKey] a where exists(select * from [Sys_MemberKey] b where a.FMachineC ...

  4. python中的字符串常量,是否支持通过下标的方式赋值

    说明: 今天在看python,通过下标获取字符串常量的字符,在想是否可以通过下标的方式赋值. 操作: 1.对字符串下标赋值 >>> text='python' >>> ...

  5. 5 -- Hibernate的基本用法 --2 1 Hibernate 下载和安装

    1. 下载Hibernate压缩包 2. 解压:文件结构 ⊙ documentation : 该路径下存放了Hibernate的相关文档,包括Hibernate的参考文档和API文档等. ⊙ lib ...

  6. 8 -- 深入使用Spring -- 3...1 Resource实现类InputStreamResource、ByteArrayResource

    8.3.1 Resource实现类------InputStreamResource:访问输入流资源的实现类.ByteArrayResource:访问字节数组资源的实现类. 5. 访问字节数组资源 ⊙ ...

  7. swift--使用URLSession异步加载图片

    NSURLConnection,在ios9.0以后被废弃,以后使用URLSession类,如下图 具体样例: self.imageV.frame = CGRect(x:,y:,width:kScree ...

  8. Python3.X如何下载安装urllib2包 ?

    python 3.X版本不需要安装urllib2包,因为urllib和urllib2包集合成在一个包了 那现在问题是: 在python3.x版本中,如何使用:urllib2.urlopen()? 答: ...

  9. 深入理解磁盘文件系统之inode

    一.inode是什么? 理解inode,要从文件储存说起. 文件储存在硬盘上,硬盘的最小存储单位叫做"扇区"(Sector).每个扇区储存512字节(相当于0.5KB). 操作系统 ...

  10. repr方法字符串输出实例对象的值

    #coding=utf-8 #repr方法字符串输出实例对象的值 class CountFromBy(object): def __init__(self, val=0, incr=1): self. ...