题目传送门

题目大意

给出一个\(n\)个点的树,每个点有点权,定义一条链的贡献为该链的点数乘上链上的权值和,求出树上所有链中的权值最大值。

\(n\le 5\times 10^4\)

思路

算是我入边分治的门的一道题吧。。。借鉴了\(\texttt{Miracle}\)巨佬的图、代码以及思路(那不就是转载么???(大雾

边分治的大概意思就是说,对于给出的一棵树,我们重新构造成一颗3度树,并且保证我们要求答案的所需性质并不会丢失。然后有一个性质:

于是我们找到重心边分治一下,递归次数就是\(\log n\)级别。于是就能在一个较优的时间复杂度内解决这个问题。

那我们如何对于原树建出一个保留所需信息的3度树呢?有两种方法,第一种就是这样:

第二种就是建两个虚儿子,然后按奇偶接儿子。

回到这道题,我们显然可以用第一种方法建树,这并不影响答案。我们发现我们边分治似乎对点的问题不是很好解决,但是两个点之间的点的个数等于边的个数+\(1\),于是我们就可以通过维护边数得到点数。

然后注意虚边的边权为\(0\)就好了,似乎也没有好讲的(手动划掉)。这里讲一个小细节,将两棵子树答案合并的时候可以满足必须要经过这条边,因为不经过的话可以在子树里面找到。不过实现起来的话,强制经过似乎比较好码。(雾

\(\texttt{Code}\)

#include <bits/stdc++.h>
using namespace std; #define Int register int
#define ll long long
#define MAXN 100005 template <typename T> inline void read (T &t){t = 0;char c = getchar();int f = 1;while (c < '0' || c > '9'){if (c == '-') f = -f;c = getchar();}while (c >= '0' && c <= '9'){t = (t << 3) + (t << 1) + c - '0';c = getchar();} t *= f;}
template <typename T,typename ... Args> inline void read (T &t,Args&... args){read (t);read (args...);}
template <typename T> inline void write (T x){if (x < 0){x = -x;putchar ('-');}if (x > 9) write (x / 10);putchar (x % 10 + '0');} int n,las[MAXN];
ll ans,val[MAXN]; namespace Graph{
#define fi first
#define se second
#define MP make_pair
#define PII pair<ll,ll>
int t,ta,tb,top = 1,to[MAXN << 1],wei[MAXN << 1],nxt[MAXN << 1],head[MAXN];
void Add_Edge (int u,int v,int w){to[++ top] = v,wei[top] = w,nxt[top] = head[u],head[u] = top;}
void AddEdge (int u,int v,int w){Add_Edge (u,v,w),Add_Edge (v,u,w);}
bool vis[MAXN << 1];
int siz[MAXN],lim,ed,Siz;//ed表示当前重心边
void dfs (int u,int fa){
siz[u] = 1;
for (Int i = head[u],v;i;i = nxt[i]){
if (vis[i] || (v = to[i]) == fa) continue;
dfs (v,u),siz[u] += siz[v];
int tmp = max (siz[v],Siz - siz[v]);
if (tmp < lim) lim = tmp,ed = i;
}
}
PII A[MAXN],B[MAXN],*f;
void dfs (int u,int fa,ll minn,ll dis){
minn = min (minn,val[u]),f[++ t] = MP (minn,dis);
for (Int i = head[u],v;i;i = nxt[i]){
if (vis[i] || (v = to[i]) == fa) continue;
dfs (v,u,minn,dis + wei[i]);
}
}
void Solve (int u,int S){//S表示当前子树大小
if (S <= 1) return ;
lim = Siz = S,dfs (u,0),vis[ed] = vis[ed ^ 1] = 1;
t = 0,f = A,dfs (to[ed],0,1e9,0),ta = t;
t = 0,f = B,dfs (to[ed ^ 1],0,1e9,0),tb = t;
sort (A + 1,A + ta + 1),sort (B + 1,B + tb + 1);
int j = tb;ll mx = 0,l = wei[ed];
for (Int i = ta;i;-- i){
while (j > 1 && B[j].fi >= A[i].fi) mx = max (mx,B[j --].se);
if (j < tb) ans = max (ans,(A[i].se + mx + 1 + l) * A[i].fi);
}
j = ta,mx = 0;
for (Int i = tb;i;-- i){
while (j > 1 && A[j].fi >= B[i].fi) mx = max (mx,A[j --].se);
if (j < ta) ans = max (ans,(B[i].se + mx + 1 + l) * B[i].fi);
}
int tx = to[ed],ty = to[ed ^ 1];
Solve (tx,siz[tx]),Solve (ty,S - siz[tx]);
}
} int top = 1,cnt,to[MAXN << 1],nxt[MAXN << 1],head[MAXN];
void Add_Edge (int u,int v){to[++ top] = v,nxt[top] = head[u],head[u] = top;} void dfs (int u,int fa){
for (Int i = head[u],v;i;i = nxt[i]){
if ((v = to[i]) == fa) continue;dfs (v,u);
if (!las[u]) Graph::AddEdge (u,v,1),las[u] = u;
else ++ cnt,Graph::AddEdge (las[u],cnt,0),Graph::AddEdge (las[u] = cnt,v,1),val[cnt] = val[u];
}
} signed main(){
read (n),cnt = n;
for (Int i = 1;i <= n;++ i) read (val[i]);
for (Int i = 2,u,v;i <= n;++ i) read (u,v),Add_Edge (u,v),Add_Edge (v,u);
dfs (1,0),Graph::Solve (1,cnt);
write (ans),putchar ('\n');
return 0;
}

题解 最长道路tree的更多相关文章

  1. 【BZOJ2870】最长道路tree 点分治+树状数组

    [BZOJ2870]最长道路tree Description H城很大,有N个路口(从1到N编号),路口之间有N-1边,使得任意两个路口都能互相到达,这些道路的长度我们视作一样.每个路口都有很多车辆来 ...

  2. BZOJ2870—最长道路tree

    最长道路tree Description H城很大,有N个路口(从1到N编号),路口之间有N-1边,使得任意两个路口都能互相到达,这些道路的长度我们视作一样.每个路口都有很多车辆来往,所以每个路口i都 ...

  3. BZOJ2870: 最长道路tree

    题解: 子树分治的做法可以戳这里:http://blog.csdn.net/iamzky/article/details/41120733 可是码量... 这里介绍另一种好写又快的方法. 我们还是一颗 ...

  4. 【bzoj2870】最长道路tree 树的直径+并查集

    题目描述 给定一棵N个点的树,求树上一条链使得链的长度乘链上所有点中的最小权值所得的积最大. 其中链长度定义为链上点的个数. 输入 第一行N 第二行N个数分别表示1~N的点权v[i] 接下来N-1行每 ...

  5. bzoj2870最长道路tree——边分治

    简化版描述: 给定一棵N个点的树,求树上一条链使得链的长度乘链上所有点中的最小权值所得的积最大. 其中链长度定义为链上点的个数.   有几个不同的做法: 1.sort+并查集+树的直径.边从大到小加入 ...

  6. BZOJ2870 最长道路tree(并查集+LCA)

    题意 (n<=50000) 题解 #include<iostream> #include<cstring> #include<cstdio> #include ...

  7. BZOJ 2870: 最长道路tree 树的直径+并查集

    挺好的一道题. 把所有点都离线下来,一个个往里加入就行了. #include <cstdio> #include <algorithm> #define N 100003 #d ...

  8. 2870: 最长道路tree

    链接 https://www.lydsy.com/JudgeOnline/problem.php?id=2870 思路 先把树转化为二叉树 再链分治 %%yyb 代码 #include <ios ...

  9. bzoj 2870 最长道路tree——边分治

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2870 关于边分治:https://www.cnblogs.com/Khada-Jhin/p/ ...

随机推荐

  1. The First Week luckyzpp

    一 ,LINUX系列有很多版本,只是我们很少去了解到更多,我们熟知红帽CentOS,Ubuntu,Debian,   Kali,  Rocky各种版本系列. 二 目前Linux 生产主流版本如下: C ...

  2. redis>lua脚本

    String lua="local num=redis.call('incr',KEYS[1])\n"+"if tonumber(num)==1 then\n" ...

  3. Mybatis原理和代码剖析

    参考资料(官方) Mybatis官方文档: https://mybatis.org/mybatis-3/ Mybatis-Parent : https://github.com/mybatis/par ...

  4. openresty lua-resty-string md5 sha aes random string

    安装 https://github.com/openresty/lua-resty-string $ sudo opm get openresty/lua-resty-string $ ls -al ...

  5. vue 打开新窗口进行打印

    父文件 let { href } = this.$router.resolve({ path: ' 自己配置本地路由,不需要动态路由 ', query: 个人建议传一整个对象 }) window.op ...

  6. vue element-ui 组件上传图片 之后 对上传按钮 进行隐藏,删除之后重新显示

    注:如果在当前的 vue 文件里 写了 style 样式,得 去除 scoped [私有属性必须去除,不能保留](这个是重点,不去除不生效), template 部分 <el-upload    ...

  7. Redis详解(三)——

    redis https://www.cnblogs.com/zhangyinhua/p/14504717.html

  8. Linux下ansible使用

    一.ansible的功能和意义 1.功能 ansible批量功能 ----------------------> 并行 01. 可以实现批量系统操作配置 02. 可以实现批量软件服务部署 03. ...

  9. Python - 面向对象编程 - 魔术方法(双下划线方法)

    什么是魔术方法 在Python中,所有以 __ 双下划线包起来的方法,都统称为 Magic Method 魔术方法,也叫双下划线方法 有哪些重要的魔术方法? __new__ https://www.c ...

  10. 运输层协议:TCP和UDP

    运输层简介 运输层的通信实体不再是主机,而是主机中的进程.运输层的通信是一台主机的进程和另一台主机的进程进行数据交换. 运输层作用 运输层向上层的应用层提供通信服务 运输层为进程提供端到端的通信 运输 ...