【[SHOI2012]随机树】
感觉第一问就非常神仙,还有第二问怎么被我当成组合数学题来做了
首先是第一问
期望具有线性性,于是深度平均值的期望等于深度和的期望值的平均
设\(dp_x\)表示具有\(x\)个叶子节点的树的深度和的期望值是多少
我们发现扩展一个叶子节点的实质将其变成两个深度原来大一的叶节点,所以对整个答案的贡献也就是这个被扩展的叶子节点的深度乘\(2\),再加上\(2\)
比如我们当前扩展的是叶子节点\(1\),那么答案就从\(dep_1+dep_2+...+dep_x\)变成了\(2*(dep_1+1)+dep_2+...+dep_x\)
\(dep_2,dep_3\)同理,也就会发现在最终的答案里每个\(dep\)都出现了\(x+1\)次
设\(dp[x]\)表示\(x\)个叶子节点的期望深度和,\(dp[x]=\sum_{i=1}^xdep_x\)
那么期望是
\]
\]
也就是
\]
最后的答案就是\(\frac{dp[n]}{n}\)了
之后第二问我就感受到了玄学的力量,各种玄学调参数
第二问好像非常麻烦的样子,没有办法像刚才那个样子从平均的角度来考虑了,而直接求期望好像不太好求,于是可以求出概率来
设\(dp[x][h]\)表示有\(x\)个叶子节点构成的树深度为\(h\)的概率是多少,那么答案就是\(\sum_{h=1}^ndp[n][h]*h\)
我们考虑一下如何求这个\(dp[n][h]\)
有一个比较套路的东西就是枚举左右子树有几个叶子节点
所以就有
\]
其中\(p[i][j]=\sum_{k=1}^jdp[i][k]\)也就是一个概率的前缀和,\(P_{i,k}\)表示一共\(i\)个叶子节点其中\(k\)个在左子树上的概率
也就是枚举左右子树的叶子节点的个数,之后对应好相应的深度,乘上这个转移发生的概率
先不考虑这个\(P_{i,k}\)怎么求,也会发现上面那个转移好像有些问题,它算重了左右两边子树的深度都是\(j-1\)的情况,于是上面还需要再减掉\(dp[k][j-1]*dp[i-k][j-1]\)
现在的问题就变成了\(P_{i,k}\)怎么求了
首先经过感性理解/手玩样例/归纳证明可以发现,在不同的扩展顺序下使得左子树上有\(k\)个叶子节点的概率是一个固定的值
我们要让左右两边共有\(i\)个叶子节点,也就是说我们一共需要扩展\(i-1\)次,第一次扩展肯定是需要扩展在当前的这个节点上的,于是还要有\(i-2\)次扩展被分给了左右子树
我们再来考虑一下使得左子树上有\(k\)个叶子节点的实质是什么,不就是分给左边的扩展次数为\(k-1\)吗,那么这样一共有\(\binom{i-2}{k-1}\)种扩展情况会使得左子树上扩展了\(k-1\)次
又因为这些不同的扩展顺序出现的概率是一样的,所以我们可以考虑一些求出这个概率
这个概率的分母上肯定是\(2*3*4*5*...*(i-1)\),因为一共需要扩展产生\(i\)个节点每次选中左子树或者右子树的概率是\(\frac{\text{左/右子树上叶子节点数量}}{\text{叶子节点的总数量}}\),而分子上由于我们在左边一共选择了\(k\)次,所以分母会有\(1*2*..*(k-1)\),也就会有相应的\(1*2*...*(i-k-1)\)
所以
\]
我们再顺便化一下柿子
\]
我下面的代码预处理了阶乘和组合数,其实直接用\(\frac{1}{i-1}\)就好了
我才不会说我看到题解才想起来继续化柿子的
于是这样转移就好了
代码
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#define re register
#define maxn 105
inline int read()
{
char c=getchar();
int x=0;
while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9')
x=(x<<3)+(x<<1)+c-48,c=getchar();
return x;
}
int opt_Q;
int n;
namespace Ask1
{
double dp[maxn];
inline void init()
{
n=read();
dp[1]=0;
for(re int i=1;i<n;i++)
dp[i+1]=((i+1)*dp[i]+2*i)/double(i);
printf("%.6lf",dp[n]/double(n));
}
}
namespace Ask2
{
double dp[maxn][maxn+100],pre[maxn][maxn];
long double fac[maxn];
long double c[maxn][maxn];
inline void init()
{
n=read();
dp[1][0]=1;dp[2][1]=1;dp[3][2]=1;
for(re int i=0;i<=n;i++) pre[1][i]=1;
for(re int i=1;i<=n;i++) pre[2][i]=1;
for(re int i=2;i<=n;i++) pre[3][i]=1;
fac[0]=1;
for(re int i=1;i<=n;i++) fac[i]=fac[i-1]*i;
c[0][0]=1;
for(re int i=1;i<=n;i++) c[i][0]=c[i][i]=1;
for(re int i=2;i<=n;i++)
for(re int j=1;j<n;j++)
c[i][j]=c[i-1][j-1]+c[i-1][j];
for(re int i=4;i<=n;i++)
for(re int j=log2(i);j<=n;j++)
{
for(re int k=1;k<i;k++)
dp[i][j]+=(dp[k][j-1]*pre[i-k][j-1]+dp[i-k][j-1]*pre[k][j-1]-dp[k][j-1]*dp[i-k][j-1])*c[i-2][k-1]*fac[k-1]*fac[i-k-1]/fac[i-1];
pre[i][j]=dp[i][j]+pre[i][j-1];
}
double ans=0;
for(re int h=log2(n);h<=n;h++)
ans+=dp[n][h]*h;
printf("%.6lf",ans);
}
}
int main()
{
opt_Q=read();
if(opt_Q==1) Ask1::init();
else Ask2::init();
return 0;
}
【[SHOI2012]随机树】的更多相关文章
- [SHOI2012]随机树
[SHOI2012]随机树 题目大意( 网址戳我! ) 随机树是一颗完全二叉树,初始状态下只有一个节点. 随机树的生成如下:每次随机选择一个叶子节点,扩展出两个儿子. 现在给定一个正整数\(n\)(\ ...
- P3830 [SHOI2012]随机树 题解
P3830 随机树 坑题,别人的题解我看了一个下午没一个看得懂的,我还是太弱了. 题目链接 P3830 [SHOI2012]随机树 题目描述 输入输出格式 输入格式: 输入仅有一行,包含两个正整数 q ...
- P3830 [SHOI2012]随机树
P3830 [SHOI2012]随机树 链接 分析: 第一问:f[i]表示有i个叶子结点的时候的平均深度,$f[i] = \frac{f[i - 1] + 2 + f[i - 1] * (i - 1) ...
- bzoj2830: [Shoi2012]随机树
题目链接 bzoj2830: [Shoi2012]随机树 题解 q1好做 设f[n]为扩展n次后的平均深度 那么\(f[n] = \frac{f[n - 1] * (n - 1) + f[n - 1] ...
- luogu P3830 [SHOI2012]随机树 期望 dp
LINK:随机树 非常经典的期望dp. 考虑第一问:设f[i]表示前i个叶子节点的期望平均深度. 因为期望具有线性性 所以可以由每个叶子节点的期望平均深度得到总体的. \(f[i]=(f[i-1]\c ...
- luogu3830 [SHOI2012]随机树
传送门:洛谷 题目大意:对于一个只有一个节点的二叉树,一次操作随机将这棵树的叶节点的下方增加两个节点.$n-1$次操作后变为$n$个叶节点的二叉树.求:(1)叶节点平均深度的期望值(2)树深度的数学期 ...
- luogu P3830 [SHOI2012]随机树
输入格式 输入仅有一行,包含两个正整数 q, n,分别表示问题编号以及叶结点的个数. 输出格式 输出仅有一行,包含一个实数 d,四舍五入精确到小数点后 6 位.如果 q = 1,则 d 表示叶结点平均 ...
- [SHOI2012]随机树[期望dp]
题意 初始 \(1\) 个节点,每次选定一个叶子节点并加入两个儿子直到叶子总数为 \(n\),问叶子节点深度和的平均值的期望以及最大叶子深度的期望. \(n\leq 100\) . 分析 对于第一问, ...
- 洛谷P3830 [SHOI2012]随机树(期望dp)
题面 luogu 题解 第一问: 设\(f[i]\)表示\(i\)步操作后,平均深度期望 \(f[i] = \frac {f[i - 1] * (i - 1)+f[i-1]+2}{i}=f[i-1]+ ...
随机推荐
- [FORWARD]ODBC 各种数据库连接串
Overview Generally, one of the first steps when you are trying to work with databases is open it. Yo ...
- ASP.NET中让图片以二进制的形式存储在数据库中
今早有个网友问到我这问题,以前我都是直接在数据库中存文件名的,还没有试过存储整张图片到数据库中,上网搜索了一下,自己又测试了一番,代码如下:建立保存图片的表的SQL语句: USE [niunantes ...
- WPF月视图控件
简介 在做一个应用时,需要做成日历月视图的形式.自己做较麻烦,于是上网找找看,在CodeProject上发现了这个Quick and Simple WPF Month-view Calendar,可是 ...
- Linux安装之后需要进行的一些步骤
查看IP 首先安装后需要查看ip用SSH或者XSHELL来连接Linux,查看ip代码 ifconfig 需要执行 sudo yum install net-tools 命令安装之后 就可以看到ip了 ...
- Spring MVC 实现Excel的导入导出功能(2:Excel的导入优化和Excel的导出)
Excel的导入V2优化版 有些时候文件上传这一步骤由前端来处理,只将上传后的 URL 传输给后端(可以参考上一文中的图片上传功能),也就是导入请求中并不会直接处理 MultipartFile 对象, ...
- SSM框架文件远程服务器下载
1.首先你必须要建立连接 获取URL的输入流 2.之后就是文件读取和写入了 3.还有就是设置响应头,响应码等 代码 @RequestMapping("/fileDownLoad") ...
- 文件上传(Servlet/Struts2/SpringMVC)
文件下载(Servlet/Struts2)的链接:http://www.cnblogs.com/ghq120/p/8328093.html 文件上传 Servlet实现 要实现文件上传的功能,必须在f ...
- shutil的一些基本用法
import shutil import time import tarfile # 将文件内容拷贝到另一个文件中 shutil.copyfileobj(open('a1', 'r'), open(' ...
- Spring cloud Zuul网关异常处理
Spring cloud Zuul网关异常处理 一 异常测试: 1> 创建一个pre类型的过滤器,并在该过滤器的run方法实现中抛出一个异常.比如下面的实现,在run方法中调用的doSometh ...
- <meta name="viewport"content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">的作用
本人对该标签理解不深,这里是复制了穆乙的文章:如果有人进来看到这篇文章,请按此https://www.cnblogs.com/pigtail/archive/2013/03/15/2961631.ht ...