1. 程式人生 > >【BZOJ3302】[Shoi2005]樹的雙中心 DFS

【BZOJ3302】[Shoi2005]樹的雙中心 DFS

ace src 表示 之間 如何 移動 sin http 我們

【BZOJ3302】[Shoi2005]樹的雙中心

Description

技術分享

Input

第一行為N,1<N<=50000,表示樹的節點數目,樹的節點從1到N編號。
接下來N-1行,每行兩個整數U,V,表示U與V之間有一條邊。
再接下N行,每行一個正整數,其中第i行的正整數表示編號為i的節點權值為W(I),樹的深度<=100

Output

將最小的S(x,y)輸出,結果保證不超過19^9

Sample Input

5
1 2
1 3
3 4
3 5
5
7
6
5
4

Sample Output

14

HINT

選取兩個中心節點為2,3

題解:這題的做法還是挺神的~

有一種暴力的方法:先枚舉一條邊,將這條邊斷開,然後兩邊分別求重心,但是這樣做復雜度有點高。

如何優化呢?觀察到樹高只有300,所以我們可以從樹根開始,不斷向靠近重心的方向移動,不能移動時就找到了重心,復雜度是樹高級別的,可以通過此題。

#include <cstdio>
#include <iostream>
#include <cstring>
using namespace std;
const int maxn=50010;
typedef long long ll;
int n,cnt;
int head[maxn],to[maxn<<1],next[maxn<<1],fa[maxn],ins[maxn];
ll siz[maxn],f[maxn][2],g[maxn];
ll tot,now,ans,sz;
inline void add(int a,int b)
{
	to[cnt]=b,next[cnt]=head[a],head[a]=cnt++;
}
inline ll rd()
{
	ll ret=0,f=1;	char gc=getchar();
	while(gc<‘0‘||gc>‘9‘)	{if(gc==‘-‘)f=-f;	gc=getchar();}
	while(gc>=‘0‘&&gc<=‘9‘)	ret=ret*10+gc-‘0‘,gc=getchar();
	return ret*f;
}
void dfs1(int x)
{
	for(int i=head[x];i!=-1;i=next[i])	if(to[i]!=fa[x])
	{
		fa[to[i]]=x,dfs1(to[i]),siz[x]+=siz[to[i]],g[x]+=g[to[i]]+siz[to[i]];
		if(siz[to[i]]>siz[f[x][0]])	f[x][1]=f[x][0],f[x][0]=to[i];
		else	f[x][1]=(siz[f[x][1]]>siz[to[i]])?f[x][1]:to[i];
	}
}
inline ll calc(int x,int y)
{
	ll sy=siz[y]-(ins[y])*sz;
	return now-2*sy+tot;
}
void dfs3(int x)
{
	ll t1=calc(x,f[x][0]),t2=calc(x,f[x][1]);
	if(t1<t2)
	{
		if(t1<now)	now=t1,dfs3(f[x][0]);
	}
	else	if(t2<now)	now=t2,dfs3(f[x][1]);
}
void dfs2(int x,int dep)
{
	ins[x]=1;
	if(x!=1)
	{
		ll tmp=0;
		tot=siz[x],sz=0,now=g[x],dfs3(x),tmp+=now;
		tot=siz[1]-siz[x],sz=siz[x],now=g[1]-g[x]-dep*siz[x],dfs3(1),tmp+=now;
		ans=min(ans,tmp);
	}
	for(int i=head[x];i!=-1;i=next[i])	if(to[i]!=fa[x])	dfs2(to[i],dep+1);
	ins[x]=0;
}
int main()
{
	n=rd();
	int i,a,b;
	memset(head,-1,sizeof(head));
	for(i=1;i<n;i++)	a=rd(),b=rd(),add(a,b),add(b,a);
	for(i=1;i<=n;i++)	siz[i]=rd();
	dfs1(1),ans=g[1],dfs2(1,0);
	printf("%lld",ans);
	return 0;
}

【BZOJ3302】[Shoi2005]樹的雙中心 DFS