#C241118A. 树


#C241118A. 树

标签(Label)

  • 树的直径

网址(Website)

题目详情 - 树 - Super

题解 - 树 - Super

题目(Problem)

样例

输入数据 1

6
1 2
1 3
2 4
2 5
3 6

输出数据 1

1 1 2 4 6 6

【样例 1 解释】

$k = 4$ 时,联通块为 ${1}, {2}, {3}, {4,5,6}$。

输入数据 2

5
1 2
2 3
3 4
3 5

输出数据 2

1 1 3 5 5

【样例 2 解释】

$k = 3$ 时,联通块为 ${1,4,5}, {2}, {3}$。

【样例 3】

见选手目录下的 $tree3.in$ 和 $tree3.ans$。

该样例满足子任务 $2$ 的限制。

【样例 4】

见选手目录下的 $tree4.in$ 和 $tree4.ans$。

该样例满足子任务 $5$ 的限制。

题解(Solution)

性质一:在树的直径上,令直径两端为 $x,y$ ,那么距离直径上的点 $i$ 距离最远的点一定是 $x$ 或 $y$ 的其中一个。

性质二:在以上的条件下,距离不在直径上的点 $j$ 最远的点一定是 $x$ 或 $y$ 的其中一个。

那么直接求出所有的点最远的距离,然后以距离为下标存入数组中,从 $n$ 到 $1$ 枚举最长距离,然后计算连通块即可。

时间复杂度:$O(n)$ 。

出题者题解

代码(Code)

#include<bits/stdc++.h>
#include<vector>
#include<queue>
#include<stack>
#define For(i,l,r) for(int i=l;i<=r;i++)
#define Rof(i,l,r) for(int i=l;i>=r;i--)
using namespace std;
#define P pair<int,int> 
#define int long long
#define x first
#define y second
inline int rd(){
	char c;bool f=false;while(!isdigit(c=getchar()))f=c=='-';int x=c^48;
	while(isdigit(c=getchar())){x=(((x<<2)+x)<<1)+(c^48);}return f?-x:x;
}
inline void write(int x){
	if(x<0) putchar('-'), write(-x);
	else if(x<10) putchar(x+'0');
	else write(x/10), putchar(x%10+'0');
}
const int inf = 0x3f3f3f3f3f3f3f3f;
const int N = 501234;
bool ST;
double Tim;

vector<int> ft[N];
int n,root1,root2,len;
int dis[N];

void dfs_dis(int x,int fa){
	dis[x] = dis[fa] + 1;
	for(auto y:ft[x]) if(y^fa) dfs_dis(y,x);
}

bitset<N> vst;
void dfs_vis(int x,int fa,int to,int step){
	if(x==to) return vst[x] = true, dis[x] = max(len-step, step), void();
	for(auto y:ft[x]) if(y^fa){
		dfs_vis(y,x,to,step+1);
		if(vst[y]) return vst[x] = true, dis[x] = max(step, len-step), void();
	}
}

void dfs_tag(int x,int fa){
	for(auto y:ft[x]) if(!vst[y]) vst[y]=true, dis[y] = dis[x]+1, dfs_tag(y,x);
}

int v[N],as[N];
int fa[N];inline int find(int x){
	if(x==fa[x]) return x;
	return fa[x] = find(fa[x]);
}

void Solve(){
	n=rd();For(i,1,n-1){
		int x=rd(),y=rd();
		ft[x].emplace_back(y);
		ft[y].emplace_back(x);
	}
	dis[0] = -1, dfs_dis(1,0);
	For(i,1,n) root1 = dis[root1]>dis[i] ? root1 : i;
	dis[0] = -1, dfs_dis(root1,0);
	For(i,1,n) root2 = dis[root2]>dis[i] ? root2 : i;
	len = dis[root2];//树的直径 
	dfs_vis(root1,0,root2,0);
	For(i,1,n) if(vst[i]) dfs_tag(i,0);
	
	For(i,1,n) v[dis[i]]++;
	int leijia = 0;
	Rof(i,n,1){
		leijia += v[i];
		as[i] = min(n-leijia+1, n);
	}For(i,1,n) write(as[i]), putchar(' ');
}

bool ED;
signed main(){
	cerr<<abs(&ST-&ED)/1024./1024.<<"MB\n";
	freopen("tree.in","r",stdin);
	freopen("tree.out","w",stdout);
	int Tt=1;Tim=clock();
	while(Tt--) Solve();
	return cerr<<"TIME:"<<(clock()-Tim)/CLOCKS_PER_SEC,0;
}

文章作者: WolfDeer
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 WolfDeer !
  目录