【AtCoder Grand Contest 018 D】Tree and Hamilton Path
贪心,树的重心。
题目大意:
给定一棵带权树,要求从一个结点出发,每次走到一个之前没有走到过的结点。所有结点都经过之后停下。要求最大化总路径长度。
题解:
贪心。我们希望最大化每条边的贡献。如果能找到一个根,使得每个结点的贡献都是它到根的距离的两倍,这样一定是最优的。
容易想到树的重心。我们以树的重心为根,每次从一个结点走到与它不在同一棵子树中的另一个结点上。这样贡献是最大的。
这样算出来是个回路,我们考虑最后在根的儿子处停下即可(即删去根连出的一条边)。
若存在一棵子树大小为 $\frac{n+1}2$,则只能去掉这棵子树与根的边。否则可以去任意,选最小的即可。
稍加推理可得把根移动以后的解不会超过当前。
Code:
#include<cstdio>
const int N=1e5+6;
typedef long long LL;
int n,head[N],cnt,nod,sz[N],mxd[N],rt;
LL dis[N];
struct edge{
int to,nxt,w;
}e[N<<1];
LL ans,mn=0x3fffffffffffffff;
void getrt(int now,int pre){
sz[now]=1;
for(int i=head[now];i;i=e[i].nxt)
if(e[i].to!=pre){
getrt(e[i].to,now);
sz[now]+=sz[e[i].to];
if(mxd[now]<sz[e[i].to])mxd[now]=sz[e[i].to];
}
if(mxd[now]<n-sz[now])mxd[now]=n-sz[now];
if(!rt||mxd[now]<mxd[rt])rt=now;
}
void dfs(int now,int pre){
sz[now]=1;
for(int i=head[now];i;i=e[i].nxt)
if(e[i].to!=pre){
dis[e[i].to]=dis[now]+e[i].w;
dfs(e[i].to,now);
sz[now]+=sz[e[i].to];
}
}
inline void update(LL x,LL y){ans+=x+y;}
int main(){
scanf("%d",&n);
for(int i=1,u,v,w;i<n;++i){
scanf("%d%d%d",&u,&v,&w);
e[++cnt]=(edge){v,head[u],w},head[u]=cnt;
e[++cnt]=(edge){u,head[v],w},head[v]=cnt;
}
getrt(1,0);
for(int i=head[rt];i;i=e[i].nxt){
dis[e[i].to]=e[i].w;
if(e[i].w<mn)mn=e[i].w;
dfs(e[i].to,rt);
}
for(int i=head[rt];i;i=e[i].nxt)if(sz[e[i].to]==(n+1)/2)mn=e[i].w;
for(int i=1;i<=n;++i)ans+=2*dis[i];
printf("%lld\n",ans-mn);
return 0;
}