NOI 2013 树的计数

NOI 2013 树的计数

洛谷传送门

题目描述

我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的 DFS 序以及 BFS 序。两棵不同的树的 DFS 序有可能相同,并且它们的 BFS 序也有可能相同,例如下面两棵树的 DFS 序都是 1 2 4 5 3,BFS 序都是 1 2 3 4 5

img

现给定一个 DFS 序和 BFS 序,我们想要知道,符合条件的有根树中,树的高度的平均值。即,假如共有 KK 棵不同的有根树具有这组 DFS 序和 BFS 序,且他们的高度分别是 h_1, h_2, \ldots, h_Kh1,h2,…,h**K,那么请你输出:

\frac{h_1+h_2+\ldots+h_K}KK**h1+h2+…+h**K

输入格式

第一行包含 11 个正整数 nn,表示树的节点个数。

第二行包含 nn 个正整数,是一个 1 \ldots n1…n 的排列,表示树的 DFS 序。

第三行包含 nn 个正整数,是一个 1 \ldots n1…n 的排列,表示树的 BFS 序。

输入保证至少存在一棵树符合给定的两个序列。

输出格式

输出 11 个实数,四舍五入保留恰好三位小数,表示树高的平均值。


题解:

一开始看到这道题还是没什么思路的。因为感觉没什么办法来通过序列来还原树。反正就是不知道咋暴力。

既然不知道暴力,就去想正解吧。没办法了。

发现,虽然序列还原树没什么思路,但是可以显然地发现一个性质:BFS序分几层,树的高度就是多少。也就是说在BFS序上枚举断点即可。

兴冲冲地去做,发现还不是那么的简单,因为情况大抵有三种:必须分,必须部分,可分可不分。

然后想想如何判断这三种情况。

首先,对于bfs序连续的两个点,不是在同一层,就是在下一层。那么这个条件可以用DFS序约束,如果BFS序连续,但是较大BFS序的DFS序反而较小,那么就一定被挪到下一层去了,这个断点位置的贡献为1.反之,贡献不一定为0,也有可能是0.5,所以确定不了。

想到这里兴高采烈了好半天,但是其实还远远未结束。

因为只考虑了DFS对BFS序的约束,BFS序必然也会对DFS序有约束。

那么,对于DFS序连续的两个点,有可能是父子关系,也有可能都是叶子,是兄弟关系。甚至有可能,是其一个点某一个祖先的儿子。

如果是兄弟关系或者父子关系,那么其BFS序也应该是连续的。也就是它们的深度差不能大于1。它们的BFS序之间最多分1层。这样的话,可以用一个差分来维护区间的这个信息。

差分维护的是分多少层。

但是如果是祖先关系,那么其BFS序就会出现逆序的情况。也就是较大DFS序的BFS序反而较小。

代码:

#include<cstdio>
using namespace std;
const int maxn=3e5+5;
int n;
int bfn[maxn],dfn[maxn],bp[maxn],dp[maxn],s[maxn];
double ans;
int main()
{
	freopen("average.in","r",stdin);
	freopen("average.out","w",stdout);
	scanf("%d",&n);
	ans=2; 
    ++s[1];
    --s[2];
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&dfn[i]);
		dp[dfn[i]]=i;
	}
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&bfn[i]);
		bp[bfn[i]]=i;
	}
	for(int i=1;i<=n;i++)
	{
		dfn[i]=bp[dfn[i]];
		bfn[i]=dp[bfn[i]];
	}
	for(int i=1;i<n;++i) 
      	if(bfn[i]>bfn[i+1]) 
        	++s[i],--s[i+1],++ans;
    for(int i=1;i<n;++i) 
      	if(dfn[i]+1<dfn[i+1]) 
        	++s[dfn[i]],--s[dfn[i+1]];
	int w=0;
    for(int i=1;i<n;++i) 
      	w+=s[i],ans+=w?0:0.5;
	printf("%.3lf\n",ans);
	return 0;
}
posted @ 2020-11-17 15:18  Seaway-Fu  阅读(123)  评论(0编辑  收藏  举报