2870: 最长道路tree
链接
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的更多相关文章
- bzoj 2870 最长道路tree——边分治
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2870 关于边分治:https://www.cnblogs.com/Khada-Jhin/p/ ...
- BZOJ 2870: 最长道路tree 树的直径+并查集
挺好的一道题. 把所有点都离线下来,一个个往里加入就行了. #include <cstdio> #include <algorithm> #define N 100003 #d ...
- 【BZOJ2870】最长道路tree 点分治+树状数组
[BZOJ2870]最长道路tree Description H城很大,有N个路口(从1到N编号),路口之间有N-1边,使得任意两个路口都能互相到达,这些道路的长度我们视作一样.每个路口都有很多车辆来 ...
- BZOJ2870—最长道路tree
最长道路tree Description H城很大,有N个路口(从1到N编号),路口之间有N-1边,使得任意两个路口都能互相到达,这些道路的长度我们视作一样.每个路口都有很多车辆来往,所以每个路口i都 ...
- 【bzoj 2870】 最长道路tree
题目 边分治 边分和点分相比就是找到一条重心边,考虑所有经过这条边的路径,之后断开这条边分成两个联通块,继续分治 由于每次分治重心是一条边,所以只会产生两个联通块,考虑两个联通块显然要比像点分那样考虑 ...
- BZOJ2870: 最长道路tree
题解: 子树分治的做法可以戳这里:http://blog.csdn.net/iamzky/article/details/41120733 可是码量... 这里介绍另一种好写又快的方法. 我们还是一颗 ...
- bzoj2870最长道路tree——边分治
简化版描述: 给定一棵N个点的树,求树上一条链使得链的长度乘链上所有点中的最小权值所得的积最大. 其中链长度定义为链上点的个数. 有几个不同的做法: 1.sort+并查集+树的直径.边从大到小加入 ...
- 【bzoj2870】最长道路tree 树的直径+并查集
题目描述 给定一棵N个点的树,求树上一条链使得链的长度乘链上所有点中的最小权值所得的积最大. 其中链长度定义为链上点的个数. 输入 第一行N 第二行N个数分别表示1~N的点权v[i] 接下来N-1行每 ...
- BZOJ2870 最长道路tree(并查集+LCA)
题意 (n<=50000) 题解 #include<iostream> #include<cstring> #include<cstdio> #include ...
随机推荐
- powerdesign、navacat、ER图、uml、类图、时序图
关于建表和生成实体以及ER图的简便方法 a:用navacat客户端生成简单的ER图,并生成建表sql,执行生成表. b:用powerdesign连接数据库,反向生成带有注释的ER图. c:用ideal ...
- X.509证书及CeritificationPath及PKCS
X.509,数字证书标准.X.509用在包含SSL/TLS在内的很多网络协议中,证书内部包含一个public key和一个identity(hostname,organization等). X.509 ...
- <9>cc.Sprite组件
1.精灵 精灵(Sprite)是Cocos系列的核心概念之一,是Cocos Creator最常用的显示图像的组件. 游戏中显示一个图片,我们就可以把这个叫做”精灵” sprite,这只是简单理解概念. ...
- WebStorage (1) 实例
实例代码 <p>本页面每5秒刷新一下,这是您第<mark></mark>次进入本页面!</p> <script> if (window.lo ...
- form的action属性值对应servlet的web.xml的url-pattern
<form action="abc">在web.xml里面<servlet><servlet-name>123</servlet-name ...
- 【转】HTTP429
转载:http://codewa.com/question/45600.html Q:How to avoid HTTP error 429 (Too Many Requests) python Q: ...
- Python并发编程之线程池/进程池--concurrent.futures模块
一.关于concurrent.futures模块 Python标准库为我们提供了threading和multiprocessing模块编写相应的多线程/多进程代码,但是当项目达到一定的规模,频繁创建/ ...
- maven 项目打可执行jar包
昨晚,突然就来了紧急任务. 验签较慢,着手优化,发来一个demo. 首先需要把该demo部署在Linux上. 该项目是maven 项目,所以用maven打个jar包,打完jar包之后, ...
- freemark、jsp&css
**************************************************************freemark相关**************************** ...
- usb通信小结
2010-07-25 16:52:00 目前了解了usb通信层面的一些基础知识如下.如果有空还要再了解hid报告描述符及协议的数据包波形. 一,USB的一些基本概念 1. 管道(Pipe) 是主机和设 ...