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 ...
随机推荐
- 解决ORA-00257: 归档程序错误。在释放之前仅限于内部连接
昨天尝试通过plsqldev尝试连接oracle数据库,报错,提示ORA-00257: 归档程序错误.在释放之前仅限于内部连接 通过查询,得知原因是archivedlog达到了数据库设置的空间限制. ...
- META标签之关键词、网页描述设置帮助SEO网站优化(转)
ASP.NET 4.0 Web Forms针对SEO改进措施中有一个是在Page类中加了2个新属性:MetaKeywords 和MetaDescription,它们使得在后台代码类中用编程的手法设 ...
- GCD(莫比乌斯+去重)
题目链接 莫比乌斯反演模板题, 去重即可: 我们可以发现只有在区间重叠部分才会有重复且为cal(e, e, k)/2;(e表示b, d中较小的一个): #include<cstdio> # ...
- css selectors tips
from https://saucelabs.com/resources/articles/selenium-tips-css-selectorsSauce Labs uses cookies to ...
- GCD(Swift)
1.取消过去的接口 说起 GCD, 大家肯定回想起类似 dispatch_async 这样的语法. GCD 的这个语法模式无论是和 Objc 还是 Swift 的整体风格都不太打掉. 所以 Swift ...
- Linux——CentOS7安装gcc编译器详解
使用yum安装gcc 使用yum命令安装还是非常easy的. yum -y install gcc gcc-c++ kernel-devel //安装gcc.c++编译器以及内核文件 手动安装gcc ...
- redis和mongodb的比较
>>RedisRedis的优点:支持多种数据结构,如 string(字符串). list(双向链表).dict(hash表).set(集合).zset(排序set).hyperloglog ...
- RN与android原生开发混合后的环境报错问题
RN与android原生开发混合后的环境报错问题 需要先安装nodejs$ yarn --version1.12.1更新当前版本yarn upgrade --latest安装 | Yarnhttps: ...
- window、linux安装jdk,excel 导入oracle,WebService,window 端口查看,svn服务安装,oracle用户解锁
内存泄露分析插件http://download.eclipse.org/mat/1.3/update-site/birt插件http://download.eclipse.org//birt/upda ...
- AtCoder Beginner Contest 084(AB)
A - New Year 题目链接:https://abc084.contest.atcoder.jp/tasks/abc084_a Time limit : 2sec / Memory limit ...