【BZOJ3302】[Shoi2005]树的双中心 DFS

【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;
}
时间: 2024-08-10 10:27:49

【BZOJ3302】[Shoi2005]树的双中心 DFS的相关文章

BZOJ3302: [Shoi2005]树的双中心

n<=50000的树,深度<=100,有点权,选两个点x,y,使最小. dis取了min之后,整个树就会以某条边为分界线分成两半,一半归一个点管.如果是两棵完全独立的树的话,那肯定分别取这两棵树的带权重心.但割掉某条边再找两边重心,这种情况不一定是合法情况.例如: 上图中,虚线边被断开,两边的重心分别是星标节点.这不是一个合法方案,但它显然不如一个合法方案的答案优: 所以放心大胆地割就好了.注意到本题中树的深度h很小,所以割边后涉及的子树信息修改操作都可以暴力修改. 把树以某点为根,希望能预处

P2726 [SHOI2005]树的双中心 题解

CSDN同步 原题链接 简要题意: 给定一棵树,\(d_{x,y}\) 为 \(x\) 与 \(y\) 距离(\(d_{x,x} = 0\)),选出两个点 \(x,y\),最小化: \[\sum_{u \in V} (w_u \times \min(dis_{x,u} , dis_{y,u})) \] 这种水的树形dp 黑题,没几个人做真是太可惜了 首先我们要明白这个式子是什么意思. \(\min (dis_{x,u} , dis_{y,u})\),就是在 \(x\) 和 \(y\) 中找到较近

【BZOJ】1146: [CTSC2008]网络管理Network(树链剖分+线段树套平衡树+二分 / dfs序+树状数组+主席树)

第一种做法(时间太感人): 这题我真的逗了,调了一下午,疯狂造数据,始终找不到错. 后来发现自己sb了,更新那里没有打id,直接套上u了.我.... 调了一下午啊!一下午的时光啊!本来说好中午A掉去学习第二种做法,噗 好吧,现在第一种做法是hld+seg+bst+二分,常数巨大,log^4级别,目前只会这种. 树剖后仍然用线段树维护dfs序区间,然后在每个区间建一颗平衡树,我用treap,(这题找最大啊,,,囧,并且要注意,这里的rank是比他大的数量,so,我们在二分时判断要判断一个范围,即要

DTU/RTU连接双中心配置说明

TCP模式双中心数据传输 实例说明 此实例是使用TCP模式将串口收到的数据转发到服务器上,将服务器发送的数据转发到串口中.在传输过程中不对数据进行加密. 配置服务器网络 1, 确认本机服务器的外网IP地址或域名 A,此实例以外网为固定IP专线   IP地址为220.160.156.233. B,如果外网为拨号上号的方式,就需要使用域名. 条件一: 通过花生壳或是其他域名解析终端绑定外网IP 条件二: 如果只是短时间的测试通信,可以直接使用当前外网IP 查询当前外网IP方法:用百度搜索“IP” 如

浅谈同城双中心的网络部署模型

企业建设数据中心时,出于灾备的考虑,会建设两个甚至多个数据中心.例如我们经常提到的"两地三中心",即同城双中心+异地中心. 同城双中心是指在同城或邻近城市建立两个可独立承担业务的数据中心,双中心具备基本相同的业务处理能力并通过高速链路实时同步数据,日常情况下可同时分担业务及管理系统的运行,并可切换运行:灾难情况下备应急切换,保证业务的持续性.异地灾备中心是指在异地的城市建立一个备份的灾备中心,用于双中心的数据备份,当双中心出现自然灾害等原因而发生故障时,异地灾备中心可以用备份数据进行业

黑科技——树剖两次dfs转一次dfs!

黑科技--树剖两次\(dfs\)转一次\(dfs\)! 重所周知,树链剖分通常是要\(dfs?\)两次的,就像这样: int Fa[N],dep[N],Sz[N],son[N]; void dfs1(int x,int pre){ Fa[x]=pre,dep[x]=dep[pre]+1; Sz[x]=1; erep(i,G,x){ int y=G.to[i]; if(y==pre)continue; dfs(y,x); Sz[x]+=Sz[y]; (Sz[y]>Sz[son[x]])&&am

树的三种DFS策略(前序、中序、后序)遍历

之前刷leetcode的时候,知道求排列组合都需要深度优先搜索(DFS), 那么前序.中序.后序遍历是什么鬼,一直傻傻的分不清楚.直到后来才知道,原来它们只是DFS的三种不同策略. N = Node(节点) L = Left(左节点) R = Right(右节点) 在深度优先搜索的时候,以Node的访问顺序,定义了三种不同的搜索策略: 前序遍历:结点 -> 左子树 -> 右子树 中序遍历:左子树-> 结点 -> 右子树 后序遍历:左子树 -> 右子树 -> 结点 ##前

树状数组+离散化+DFS序+离线 HDOJ 4358 Boring counting

题目传送门 题意:给你一棵树,树上的每个节点都有树值,给m个查询,问以每个点u为根的子树下有多少种权值恰好出现k次. 分析:首先要对权值离散化,然后要将树形转换为线形,配上图: 收获: //还没写完... 代码: /************************************************ * Author :Running_Time * Created Time :2015/9/10 星期四 19:15:22 * File Name :I.cpp ************

Codeforces 383C . Propagating tree【树状数组,dfs】

题目大意: 有一棵树,对这个树有两种操作:1:表示为(1 x val),在编号为x的节点上加上val,然后给x节点的每个儿子加上- val,再给每个儿子的儿子加上-(- val),一直加到没有儿子为止.2:表示为(2 x)查询x节点上的值. 做法: 由于每次修改操作修改的并不是一个值,而是很多值,那我们将该题抽象成区间修改,点查询的问题.那怎么抽象呢?可以明白的是,每次操作虽然有加有减,但是每次做加法操作,或者减法操作的都是同一部分数(也就是说,在某次加上同一个数的节点们,下次操作一定是加上或者