3244: [Noi2013]树的计数 - BZOJ

Description

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

现给定一个DFS序和BFS序,我们想要知道,符合条件的有根树中,树的高度的平均值。即,假如共有K棵不同的有根树具有这组DFS序和BFS序,且他们的高度分别是h1,h2,...,hk,那么请你输出
(h1+h2..+hk)/k
Input

有3行。
第一行包含1个正整数n,表示树的节点个数。
第二行包含n个正整数,是一个1~n的排列,表示树的DFS序。
第三行包含n个正整数,是一个1~n的排列,表示树的BFS序。
输入保证至少存在一棵树符合给定的两个序列。
Output

仅包含1个实数,四舍五入保留恰好三位小数,表示树高的平均值。
Sample Input

5
1 2 4 5 3
1 2 3 4 5

Sample Output
3.500
HINT

【评分方式】

如果输出文件的答案与标准输出的差不超过0.001,则将获得该测试点上的分数,否则不得分。

【数据规模和约定】

20%的测试数据,满足:n≤10;

40%的测试数据,满足:n≤100;

85%的测试数据,满足:n≤2000;

100%的测试数据,满足:2≤n≤200000。

【说明】

树的高度:一棵有根树如果只包含一个根节点,那么它的高度为1。否则,它的高度为根节点的所有子树的高度的最大值加1。

对于树中任意的三个节点a , b , c ,如果a, b都是c的儿子,则a, b在BFS序中和DFS序中的相对前后位置是一致的,即要么a都在b的前方,要么a都在b的后方。

Orz两位神犇的题解http://www.cnblogs.com/g-word/p/3288675.html

http://www.cnblogs.com/lazycal/p/bzoj-3244.html

我太弱了,你们还是看他们两个的吧,他们讲的还是比较清楚的

 1 const
 2     maxn=200200;
 3 type
 4     node=record
 5         lc,rc,l,r,min:longint;
 6     end;
 7 var
 8     a,b,dfs,bfs,max:array[0..maxn]of longint;
 9     f:array[0..maxn*2]of node;
10     n,tot:longint;
11     ans,s:double;
12
13 function min(x,y:longint):longint;
14 begin
15     if x<y then exit(x);
16     exit(y);
17 end;
18
19 procedure build(l,r:longint);
20 var
21     now,mid:longint;
22 begin
23     inc(tot);now:=tot;
24     f[now].l:=l;f[now].r:=r;
25     if l=r then
26     begin
27         f[now].min:=a[l];
28         exit;
29     end;
30     mid:=(l+r)>>1;
31     f[now].lc:=tot+1;
32     build(l,mid);
33     f[now].rc:=tot+1;
34     build(mid+1,r);
35     f[now].min:=min(f[f[now].lc].min,f[f[now].rc].min);
36 end;
37
38 function min(now,l,r:longint):longint;
39 var
40     mid:longint;
41 begin
42     if (f[now].l>=l) and (f[now].r<=r) then exit(f[now].min);
43     mid:=(f[now].l+f[now].r)>>1;
44     min:=n;
45     if l<=mid then min:=min(min(f[now].lc,l,r),min);
46     if r>mid then min:=min(min(f[now].rc,l,r),min);
47 end;
48
49 procedure main;
50 var
51     i:longint;
52 begin
53     read(n);
54     for i:=1 to n do read(a[i]);
55     for i:=1 to n do read(b[i]);
56     for i:=1 to n do bfs[b[i]]:=i;
57     for i:=1 to n do a[i]:=bfs[a[i]];
58     for i:=1 to n do dfs[a[i]]:=i;
59     for i:=1 to n do
60         if a[i]>max[i-1] then max[i]:=a[i]
61         else max[i]:=max[i-1];
62     ans:=1;
63     build(1,n);
64     for i:=1 to n-1 do
65         if (i=1) or (dfs[i+1]<dfs[i]) then ans:=ans+1+s
66         else
67             if dfs[i+1]=dfs[i]+1 then
68                 begin
69                     if max[dfs[i]]<=i+1 then s:=s+0.5;
70                 end
71             else
72                 if min(1,dfs[i],dfs[i+1])<i then s:=0;
73     ans:=ans+s;
74     writeln(ans-0.001:0:3);writeln(ans:0:3);writeln(ans+0.001:0:3);
75 end;
76
77 begin
78     main;
79 end.

3244: [Noi2013]树的计数 - BZOJ

时间: 2024-08-10 17:20:22

3244: [Noi2013]树的计数 - BZOJ的相关文章

[BZOJ3244][NOI2013]树的计数

这题大家为什么都写O(NlogN)的算法呢?…… 让本蒟蒻来写一个O(N)的吧…… 首先还是对BFS序和DFS序重编号,记标好的DFS序为d[1..n].令pos[x]为x在d[]中出现的位置,即pos[d[i]]=i. 然后还是要用到一个BFS序的分段对应一棵树的结论……然后我们考察一个分段方式的合法性:首先结点1是唯一的根必须要单独一段:其次,BFS序中一层的结点出现的顺序和DFS序中的顺序一定是相同的,因此对于任何的一段[l, r],都有pos[l]<pos[l+1]<pos[l+2]&

[bzoj3244] [洛谷P1232] [Noi2013] 树的计数

Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同,例如下面两棵树的DFS序都是1 2 4 5 3,BFS序都是1 2 3 4 5 现给定一个DFS序和BFS序,我们想要知道,符合条件的有根树中,树的高度的平均值.即,假如共有K棵不同的有根树具有这组DFS序和BFS序,且他们的高度分别是h1,h2,...,hk,那么请你输出 (h1+h2..+hk)

[NOI2013]树的计数

题目描述 能评测了哦. 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同,例如下面两棵树的DFS序都是1 2 4 5 3,BFS序都是1 2 3 4 5 现给定一个DFS序和BFS序,我们想要知道,符合条件的有根树中,树的高度的平均值.即,假如共有K棵不同的有根树具有这组DFS序和BFS序,且他们的高度分别是h1,h2,...,hk,那么请你输出 (h1+h2..+hk)

【uoj122】 NOI2013—树的计数

http://uoj.ac/problem/122 (题目链接) 题意 给出一棵树的dfs序和bfs序,保证一定可以构成一棵树.问构成的树的期望深度. Solution 这是一个悲伤的故事,我YY的东西挂了,最后打满了补丁,化简一下,就是跟llg一样的写法.←_←别理他. 本来很简单的一个东西,也许是我脑洞太大了→_→ 代码 // uoj122 #include<algorithm> #include<iostream> #include<cstdlib> #inclu

BZOJ3244 [Noi2013]树的计数/UOJ122

本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/转载请注明出处,侵权必究,保留最终解释权! Description 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同,例如下面两棵树的DFS序都是1 2 4 5 3,BFS序

【BZOJ3244】【UOJ#122】【NOI2013]树的计数

NOI都是酱的题怎么玩啊,哇.jpg 原题: 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的DFS序以及BFS序.两棵不同的树的DFS序有可能相同,并且它们的BFS序也有可能相同,例如下面两棵树的DFS序都是1 2 4 5 3,BFS序都是1 2 3 4 5 现给定一个DFS序和BFS序,我们想要知道,符合条件的有根树中,树的高度的平均值.即,假如共有K棵不同的有根树具有这组DFS序和BFS序,且他们的高度分别是h1,h2,...,hk,那么请你输出(h

UOJ#122【NOI2013】树的计数

[NOI2013]树的计数 链接:http://uoj.ac/problem/122 按BFS序来,如果$B_i$与$B_{i-1}$必须在同一层,那么贡献为0,必须在不同层那么贡献为1,都可以贡献为0.5. 因为$B_i$与$B_{i-1}$相邻,所以对方案数的改变最多+1. 必须在不同层,即$D(B_{i-1})>D(B_i)$ 都可以,$B_i$能往下移一层,不改变BFS序以及DFS序: 作为兄弟,父亲必须一样(即$D(B_{i-1})==D(B_i)-1$),不然会改变DFS序. 作为儿

【BZOJ】【1211】【HNOI2004】树的计数

Prufer序列+组合数学 嗯哼~给定每个点的度数!求树的种数!那么很自然的就想到是用prufer序列啦~(不知道prufer序列的……自己再找找资料吧,这里就不放了,可以去做一下BZOJ1005明明的烦恼) 那么我们令每个点的度数v[i]-1,得到每个节点在prufer序中的出现次数! 现在就是求这个prufer序有多少种了……有两种做法: 1.多重集排列数:n个元素,每种元素有a[i]个,求全排列的方案数,自己随便yy一下就可以得到$$ans=\frac{n!}{\prod (a[i]!)}

UOJ #122 【NOI2013】 树的计数

题目链接:树的计数 这道题好神啊--正好有人讲了这道题,那么我就写掉吧-- 首先,为了方便考虑,我们可以把节点重标号,使得\(bfs\)序变成\(1,2,3,--,n\),那么显然树的深度就是\(dep_n\). 然后,我们来考虑一下\(dfs\)序和\(bfs\)序的性质.设\(dfs\)序中的第\(i\)个点为\(d_i\),那么显然有\(dep_{d_{i+1}} \le dep_{d_i}+1\).由于我们已经进行了重标号,那么通过\(bfs\)序,我们可以得到\(dep_i \le d