zoukankan      html  css  js  c++  java
  • 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
    ",ans);
    	return 0;
    }
    
  • 相关阅读:
    剔除数组中重复对象
    2018湖南省高中数学联赛(A)试题
    向量与三角
    阿波罗尼斯圆
    2019高考理科数学(天津卷)
    转载Android常见错误(http://www.cnblogs.com/oklanyan/archive/2011/10/12/2208378.html)
    TCP/IP网络层谜云
    bitmap 设置图片尺寸,避免 内存溢出 OutOfMemoryError的优化方法(转载:http://tonyyu.iteye.com/blog/713256)
    ArcGIS for Android 离线数据编辑原理
    Android环境搭建(XP/Win7)转载http://www.cnblogs.com/oklanyan/archive/2011/10/14/2207339.html
  • 原文地址:https://www.cnblogs.com/fusiwei/p/13994350.html
Copyright © 2011-2022 走看看