[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)/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的后方。
想法##
果然noi的题就是有水准啊666
%了各方题解,琢磨了3天才大概明白了这道题
首先将每个点重新编号使bfs为1 2 3 …,dfs对应一下,这样到后面会方便很多。
然后利用期望的线性性质,最终树的高度的期望值便为每个点对树的高度的贡献的期望值之和。
设c[i]为编号为i的点在dfs序中的位置,每个点的期望贡献值为p[i]
按照bfs的顺序考虑每个点(也就是重新编号后1~n的顺序)
有下面这几种情况:
1.c[i] < c[i-1]
那么i只能在i-1的下一行,故p[i]=1
2.c[i] > c[i-1]
那么i既有可能与i-1同行,也有可能在i-1下一行
可在i-1下一行的情况如下图所示:
我们可以发现,蓝色的两个点必须满足在dfs序中相邻,且不会被其它点限制必须在同一行
会不会被两个点限制在同一行其实就是dfs序提供的信息了。
考虑dfs序中相邻的两点i,j(c[j]>c[i]),如果j>i,那么j要么是i的兄弟,要么是i的儿子
也就是说,i+1~j中所有点对树高度的贡献 \(\leq\) 1
刚刚上面所说的“被限制在同一行”其实就是这个东西。
对于每个限制,如果那一段区间内有确定的p[i]=1的点,那么其它点p[i']=0
都打上差分标记,最终没有被打标记且p[i]!=1的点就是既可作兄弟又可作儿子的点,p[i]=0.5
代码##
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 200005;
int n;
int a[N],b[N],c[N],p[N],sum[N],zz[N]; //zz差分
int main()
{
int x,s;
scanf("%d",&n);
for(int i=1;i<=n;i++) {
scanf("%d",&x);
p[x]=i;
}
for(int i=1;i<=n;i++){
scanf("%d",&x);
b[i]=i; a[p[x]]=i; c[i]=p[x];
}
if(n==1) { printf("1\n"); return 0; }
p[1]=p[2]=1;
sum[1]=1; sum[2]=2;
for(int i=3;i<=n;i++){
p[i]=0;
if(c[i]<c[i-1]) p[i]=1;
sum[i]=sum[i-1]+p[i];
}
for(int i=2;i<=n;i++)
if(a[i]>a[i-1])
if(sum[a[i]]-sum[a[i-1]]) zz[a[i-1]+1]++,zz[a[i]+1]--;
double ans=0; s=0;
for(int i=1;i<=n;i++){
s+=zz[i];
if(p[i]) ans+=1;
else if(s==0) ans+=0.5;
}
printf("%.3lf\n%.3lf\n%.3lf\n",ans-0.001,ans,ans+0.001); //bzoj神奇的输出格式
return 0;
}
[bzoj3244] [洛谷P1232] [Noi2013] 树的计数的更多相关文章
- 洛谷 P2290 [HNOI2004]树的计数
题目描述 输入输出格式 输入格式: 输入文件第一行是一个正整数n,表示树有n个结点.第二行有n个数,第i个数表示di,即树的第i个结点的度数.其中1<=n<=150,输入数据保证满足条件的 ...
- luogu P1232 [NOI2013]树的计数
传送门 这题妙蛙 首先考虑构造出一个合法的树.先重新编号,把bfs序整成\(1,2,3...n\),然后bfs序就是按照从上到下从左往右的遍历顺序,所以可以考虑对bfs序分层,可以知道分层方式只会对应 ...
- 洛谷1087 FBI树 解题报告
洛谷1087 FBI树 本题地址:http://www.luogu.org/problem/show?pid=1087 题目描述 我们可以把由“0”和“1”组成的字符串分为三类:全“0”串称为B串,全 ...
- [UOJ#122][NOI2013]树的计数
[UOJ#122][NOI2013]树的计数 试题描述 我们知道一棵有根树可以进行深度优先遍历(DFS)以及广度优先遍历(BFS)来生成这棵树的 DFS 序以及 BFS 序.两棵不同的树的 DFS 序 ...
- 洛谷P3018 [USACO11MAR]树装饰Tree Decoration
洛谷P3018 [USACO11MAR]树装饰Tree Decoration树形DP 因为要求最小,我们就贪心地用每个子树中的最小cost来支付就行了 #include <bits/stdc++ ...
- 洛谷 P3714 - [BJOI2017]树的难题(点分治)
洛谷题面传送门 咦?鸽子 tzc 竟然来补题解了?incredible( 首先看到这样类似于路径统计的问题我们可以非常自然地想到点分治.每次我们找出每个连通块的重心 \(x\) 然后以 \(x\) 为 ...
- NOIP2017提高组Day2T3 列队 洛谷P3960 线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/9265380.html 题目传送门 - 洛谷P3960 题目传送门 - LOJ#2319 题目传送门 - Vij ...
- 洛谷P3703 [SDOI2017]树点涂色(LCT,dfn序,线段树,倍增LCA)
洛谷题目传送门 闲话 这是所有LCT题目中的一个异类. 之所以认为是LCT题目,是因为本题思路的瓶颈就在于如何去维护同颜色的点的集合. 只不过做着做着,感觉后来的思路(dfn序,线段树,LCA)似乎要 ...
- 洛谷P3372线段树1
难以平复鸡冻的心情,虽然可能在大佬眼里这是水题,但对蒟蒻的我来说这是个巨大的突破(谢谢我最亲爱的lp陪我写完,给我力量).网上关于线段树的题解都很玄学,包括李煜东的<算法竞赛进阶指南>中的 ...
随机推荐
- POJ 3660 Cow Contest(floyed运用)
Description N (1 ≤ N ≤ 100) cows, conveniently numbered 1..N, are participating in a programming con ...
- H3C配置历史命令缓冲大小--接口视图(console为准)
[wang]user-interface console 0 [wang-ui-console0]history-command max-size 30 //配置缓冲区大小 [wang-ui-c ...
- Java中大量if...else语句的消除替代方案
在我们平时的开发过程中,经常可能会出现大量If else的场景,代码显的很臃肿,非常不优雅.那我们又没有办法处理呢? 针对大量的if嵌套让代码的复杂性增高而且难以维护.本文将介绍多种解决方案. 案例 ...
- Winform C#关于utf8编码问题
public string SendDataByPost(string param, string Url) { try { HttpWebRequest request = (HttpWebRequ ...
- python 实现整数的反转:给定一个整数,将该数按位逆置,例如给定12345变成54321,12320变成2321.
给定一个n位(不超过10)的整数,将该数按位逆置,例如给定12345变成54321,12320变成2321. # 第一种方法,使用lstrip函数去反转后,数字前面的0 import math num ...
- Java中的循环结构
1.while循环结构 语法: while(循环条件){ //循环操作 } while循环结构流程图: 举例: int i = 1; while(i <= 100){ System.out.pr ...
- 将 Sidecar 容器带入新的阶段
作者 | 徐迪.张晓宇 导读:本文根据徐迪和张晓宇在 KubeCon NA 2019 大会分享整理.分享将会从以下几个方面进行切入:首先会简单介绍一下什么是 Sidecar 容器:其次,会分享几个阿里 ...
- jquery中获取当前选中行数据的方法
$("table tr").click(function() { var td = $(this).find("td");// 找到td元素 var lo_id ...
- Logback 学习指南 一
因为项目中用到 SpringBoot,看到官方文档中提及默认的日志实现是 logback,因此就通过阅读手册和结合实践学习了下相关的知识,记录下以备查阅. 1. logback 是什么? logbac ...
- BeetleX之XRPC使用详解
XRPC是基于BeetleX扩展一个远程接口调用组件,它提供基于接口的方式来实现远程服务调用,在应用上非常简便.组件提供.NETCore2.1和.NETStandard2.0的client版本,因此即 ...