传送门

题意

有n个小镇,Bobo想要建造n-1条边,并且如果在u到v建边,那么花费是u到v的最短路长度(原图),问你最大的花费。

分析

比赛的时候没做出来,QAQ

我们首先要找到树的直径起点和终点,方法是

1.任意选一个点,dfs找到最长路的终点

2.从终点反向dfs,找到起点

然后枚举每个点,用倍增lca求出该点到起点与终点距离,取距离大的,注意直径本身会被访问两次,故一开始ans要减去最长路,时间复杂度O(nlogn)

代码

#include<cstdio>
#include<cstring>
#include<vector>
#include<algorithm>
using namespace std; #define ll long long
#define F(i,a,b) for(int i=a;i<=b;++i)
#define R(i,a,b) for(int i=a;i<b;++i)
#define mem(a,b) memset(a,b,sizeof(a))
#pragma comment(linker, "/STACK:102400000,102400000")
inline void read(int &x){x=0; char ch=getchar();while(ch<'0') ch=getchar();while(ch>='0'){x=x*10+ch-48; ch=getchar();}} int t,n,m;
int fa[100100],depth[100100],up[100100][20],vis[100100];
ll dis[100100],dis2[100100];
struct node
{
int to;
ll w;
node(int tt,ll ww):to(tt),w(ww){}
};
vector<node>edge[100100]; void dfs(int u,int pre,int dep)//第一次dfs,标出每个点的父亲,计算从该点到1的距离dus[i]
{
depth[u]=dep;//记录深度
fa[u]=pre;//记录父亲
int num=edge[u].size();
R(i,0,num)
{
int v=edge[u][i].to;
if(v!=pre)
{
dis[v]=dis[u]+edge[u][i].w;//更新距离
dfs(v,u,dep+1);
}
}
}
void dfs2(int x)//第二次遍历,从第一次dfs的最大距离点遍历
{
int num=edge[x].size();
vis[x]=1;//用vis记录该点是否被访问
R(i,0,num)
{
int v=edge[x][i].to;
if(!vis[v])
{
dis2[v]=dis2[x]+edge[x][i].w;//记录距离
dfs2(v);
}
}
}
void create()//构建倍增数组
{
mem(up,0);
F(i,1,n) up[i][0]=fa[i];
for(int j=1;j<20;++j)for(int i=1;i<=n;++i)
up[i][j]=up[up[i][j-1]][j-1];
}
int lca(int u,int v)//找到u和v的的最近公共祖先
{
if(depth[u]<depth[v]) swap(u,v);
int ret=depth[u]-depth[v];
R(i,0,20)if(ret&(1<<i)) u=up[u][i];//将u调整到与v同深度
if(u==v) return u;//同一侧,则返回
for(int i=19;i>=0;--i) if(up[u][i]!=up[v][i])//不同则让u和v向上跳,直到跳不了
{
u=up[u][i];
v=up[v][i];
}
return fa[u];
}
int main()
{
while(scanf("%d",&n)!=EOF)
{
int u,v;ll w;
F(i,0,n) edge[i].clear();
mem(fa,0);mem(depth,0);
mem(dis,0);mem(dis2,0);//注意dis和dis2的清空
for(int i=1;i<n;++i)
{
scanf("%d%d%I64d",&u,&v,&w);
edge[u].push_back(node(v,w));
edge[v].push_back(node(u,w));
}
dfs(1,0,0);
ll maxx=0;int End,Begin;
F(i,1,n) if(maxx<dis[i]) { maxx=dis[i];End=i; }//记录从1开始的最长路终点
create();
mem(vis,0);
dfs2(End);
maxx=0;
F(i,1,n) if(maxx<dis2[i]) { maxx=dis2[i];Begin=i; }//找到反向dfs最长路的点,那么Begin和End构成树的直径
ll ans=-maxx;//要遍历树的直径两次,故先减去一次
F(i,1,n)
{
ll ret1,ret2;
ret1=dis[i]+dis[Begin]-2*dis[lca(i,Begin)];
ret2=dis[i]+dis[End]-2*dis[lca(i,End)];
ans+=max(ret1,ret2);//去当前点到直径两端的最短路长度的最大值
}
printf("%I64d\n",ans);
}
return 0;
}

最新文章

  1. [CF442A] Borya and Hanabi (暴力bitmask)
  2. Mark一下,Android ListView的上下间隙
  3. centos6.4添加fedora-epel源
  4. vs2008编译boost
  5. R语言 典型相关分析
  6. 关于typedef int(*lpAddFun)(int, int)
  7. Css3案例
  8. 安装windows7和ubuntu双系统后引导项设置
  9. Aforge.net 一个专门为开发者和研究者基于C#框架设计
  10. 在一个form里边同时执行搜索和 execl导出功能
  11. Read程序员的困境有感
  12. xin
  13. Element link doesn&#39;t have required attribute property
  14. 洛谷.1501.[国家集训队]Tree II(LCT)
  15. jdbc --- javabean
  16. 20170908工作日记--UML画类图、HTTP协议、Volley源码走读
  17. Golang context包解读
  18. redis分布式(主从复制)
  19. Cg入门21:Fragment shader - 2D纹理採样
  20. 《DSP using MATLAB》示例 Example 9.6

热门文章

  1. Centos 6.x 安装Python 3.4.3
  2. [Testing] Static Analysis Testing JavaScript Applications
  3. 分布式搜索elasticsearch 基本概念
  4. UVA 11246 - K-Multiple Free set(数论推理)
  5. 中科燕园GIS外包---交通运输综合地理信息平台
  6. 凝视转换(c转换为c++)
  7. 使用virtualenv, uwsgi, nginx来布署flask
  8. http协议的队首阻塞
  9. Hadoop spark mongo复制集
  10. Delphi通过POST传递参数给PHP