链接

https://www.lydsy.com/JudgeOnline/problem.php?id=2870

思路

先把树转化为二叉树

再链分治

%%yyb

代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <utility>
#include <cstdio>
#include <vector>
#define ll long long
using namespace std;
const int N=4e5+7,inf=0x3f3f3f3f;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,sdgzy,w[N];
ll ans;
struct node {
int v,nxt,w;
}e[N];
int head[N],mvp;
void Add(int u,int v,int q) {
e[++mvp].v=v;
e[mvp].w=q;
e[mvp].nxt=head[u];
head[u]=mvp;
}
vector<int> G[N];
void dfs(int u,int f) {
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(v==f) continue;
G[u].push_back(v);
dfs(v,u);
}
}
void rebuild() {
mvp=1,memset(head,0,sizeof(head));
for(int i=1;i<=n;++i) {
int tmp=G[i].size();
if(tmp<=2) {
for(int j=0;j<tmp;++j) Add(i,G[i][j],G[i][j]<=sdgzy),Add(G[i][j],i,G[i][j]<=sdgzy);
} else {
int a=++n,b=++n;w[a]=w[b]=w[i];
Add(a,i,0),Add(i,a,0),Add(b,i,0),Add(i,b,0);
for(int j=0;j<tmp;++j) {
if(j&1) G[a].push_back(G[i][j]);
else G[b].push_back(G[i][j]);
}
}
}
}
bool vis[N];
int rt,rt_val,siz[N];
void get_rt(int u,int f,int tot) {
siz[u]=1;
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(vis[i>>1]||v==f) continue;
get_rt(v,u,tot);
siz[u]+=siz[v];
int tmp=max(siz[v],tot-siz[v]);
if(tmp<rt_val) rt_val=tmp,rt=i;
}
}
int top[2];
pair<int,int> SX[2][N];
void get_dis(int u,int f,int mi,int dep,int opt) {
mi=min(mi,w[u]);
SX[opt][++top[opt]]=make_pair(mi,dep);
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(vis[i>>1]||v==f) continue;
get_dis(v,u,mi,dep+e[i].w,opt);
}
}
bool cmp(pair<int,int> a,pair<int,int> b) {return a>b;}
void solve(int u,int tot) {
rt_val=inf;
get_rt(u,0,tot);
if(rt_val >= inf) return;
vis[rt>>1]=1;
ll ls=e[rt].v,rs=e[rt^1].v;//bian fen zhi
top[0]=top[1]=0;
get_dis(ls,u,inf,0,0);
get_dis(rs,u,inf,0,1);
sort(SX[0]+1,SX[0]+1+top[0],cmp);
sort(SX[1]+1,SX[1]+1+top[1],cmp);
int ma=0;
for(int i=1,j=1;i<=top[0];++i) {
while(j<=top[1]&&SX[1][j].first>=SX[0][i].first) ma=max(ma,SX[1][j].second),j++;
if(j-1<=top[1]&&SX[1][j-1].first>=SX[0][i].first)
ans=max(ans,1LL*SX[0][i].first*(SX[0][i].second+ma+1+e[rt].w));//dian = bian + 1
// if(j>top[1]) break;
}
ma=0;
for(int i=1,j=1;i<=top[1];++i) {
while(j<=top[0]&&SX[0][j].first>=SX[1][i].first) ma=max(ma,SX[0][j].second),j++;
if(j-1<=top[0]&&SX[0][j-1].first>=SX[1][i].first)
ans=max(ans,1LL*SX[1][i].first*(SX[1][i].second+ma+1+e[rt].w));
// if(j>top[0]) break;
}
int a=siz[ls],b=tot-a;
solve(ls,a);
solve(rs,b);
}
int main() {
// freopen("a.in","r",stdin);
// freopen("a.out","w",stdout);
n=read();sdgzy=n;
for(int i=1;i<=n;++i) w[i]=read();
for(int i=1;i<n;++i) {
int x=read(),y=read();
Add(x,y,0),Add(y,x,0);
}
dfs(1,0);
rebuild();
solve(1,n);
cout<<ans<< "\n";
return 0;
}

2870: 最长道路tree的更多相关文章

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

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

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

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

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

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

  4. BZOJ2870—最长道路tree

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

  5. 【bzoj 2870】 最长道路tree

    题目 边分治 边分和点分相比就是找到一条重心边,考虑所有经过这条边的路径,之后断开这条边分成两个联通块,继续分治 由于每次分治重心是一条边,所以只会产生两个联通块,考虑两个联通块显然要比像点分那样考虑 ...

  6. BZOJ2870: 最长道路tree

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

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

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

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

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

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

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

随机推荐

  1. fiddler2抓包数据工具使用教程

    一款免费且功能强大的数据包抓取软件.它通过代理的方式获取程序http通讯的数据,可以用其检测网页和服务器的交互情况,能够记录所有客户端和服务器间的http请求,支持监视.设置断点.甚至修改输入输出数据 ...

  2. 关于oracle中的数字类型

    1.关于number类型. 以下是从其文档中摘录出的一句话: p is the precision, or the total number of significant decimal digits ...

  3. executing in nfs will not generate core dump file

    最近遇到了一个奇怪的问题. linux系统的pc搭建nfs server,开发板作为nfs client,开发板中全程root权限操作,执行的程序放到 nfs server 中 exports 出的目 ...

  4. Unity shader学习之Blinn-Phong光照模型

    Blinn-Phong光照模型不用计算反射方向,计算公式如下: h = normalize(v + l); Cspecular = Clight * mspecular * pow(max(0, do ...

  5. 添加Google搜索

    给自己的网站加一个搜索功能很简单,代码如下: 欢迎访问我的博客:http://67566894.iteye.com/ <form action="http://www.google.c ...

  6. Day9 面向对象高级

    一.方法 方法包括:普通方法.静态方法和类方法,三种方法在内存中都归属于类,区别在于调用方式不同. 普通方法:由对象调用:至少一个self参数:执行普通方法时,自动将调用该方法的对象赋值给self: ...

  7. GameObject.Find与Transform.Find的区别

    1.GameObject.Find 函数原型: public static GameObject Find(string name); 说明:1.GameObject只能查找到active的物体 2. ...

  8. 删除(unfork)github中某个库(repository)

    1.头像->your profile 2.点击某个想要删除的库,出现界面: 3.点击setting,鼠标滚动页面至末尾,出现页面: 4.点击delete this repository: OK!

  9. html5-css边框img

    div{    width: 500px;    height: 300px;    background: rgb(122,30,60);    border:15px solid black;   ...

  10. 51ll网产品信息保存为txt文件

    import requests from pyquery import PyQuery as pq url='http://www.51xxx.com/Try/index/p/3' headers={ ...