题目链接:迷失游乐园

  这道题也没有传说中的那么难写吗→_→

  似乎有篇博客讲得特详细……附上链接:戳这里

  如果这道题不是基环树,而就是一棵树的话,我们来考虑改怎么做。因为树上的路径只有向上、向下两种走法,于是我们可以记\(down_u\)表示从\(u\)往下走的期望长度,\(up_u\)表示从\(u\)往上走的期望长度,这就很好转移了。

  显然对于一个点\(u\),设它的父亲为\(fa_u\),有\(son_u\)个孩子和\(num_u\)个父亲(根节点认为没有父亲),那么显然有:

\[down_u=\frac{\sum_{fa_v=u}down_v+dis(u,v)}{son_u}\]

  这就是往各个孩子走的概率相等,于是求个平均值就可以了。

  然后我们接着考虑\(up_u\)怎么求:

\[up_u=dis(fa_u,u)+\frac{up_{fa_u}\times num_u+down_{fa_u}\times son_{fa_u}-down_u-dis(fa_u,u)}{son_{fa_u}-1+num_{fa_u}}\]

  因为我们下一步从\(u\)走到了\(fa_u\),那么之后一步可以往下也可以向上。往上走的总和就是\(up_{fa_u}\times num_u\),往下走的话由于不能走回头路,于是就用\(down_{fa_u}\times son_{fa_u}\)减去从\(u\)往下的\(down_u\)和从\(fa_u\)走过来的\(dis(fa_u,u)\)即可。由于期望相当于平均值,共有\(son_{fa_u}-1+num_{fa_u}\)种种情况,总和除以总情况数就是期望。

  于是树的情况就做完辣!接下来让我们来考虑基环树的情况。

  我们可以把基环树看成一个环上面长着许多棵大大小小的树。在这种情况下,我们发现每棵树的根节点也可以往上走了,并且在环上有两种走法。为了方便接下来的计算,我们可以先把环上每个点的\(down\)值给求出来。我们可以对于每个点,枚举它顺时针走到哪个点,那么在这个点往它的子树走的概率我们是可以递推算出来的,并且由于我们知道了每个点的\(down\)值,所以知道会向下走多远,于是就可以算出顺时针走的期望了。逆时针同理。于是我们就在\(O(环长^2)\)的时间内求出环上每个点的\(up\)值了。剩下的部分就是树的情况了,只不过对于每个环上的点\(x\)有\(num_x=2\),因为有往上有两种走法。

  说了这么多,最后的答案就是:\[\sum_{i=1}^{n}\frac{up_i\times num_i+down_i\times son_i}{num_i+son_i}\]

  代码也不长:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
#define maxn 100010 using namespace std;
typedef double llg; int n,m,du[maxn],dis[maxn],hf[maxn];//hf[u]表示u有几个父亲,环上的点认为有两个父亲
int dfn[maxn],a[maxn],la,b[maxn],fa[maxn];//a数组中存的是环上的点,b[i]存的是a[i]与a[i-1]之间的距离
int head[maxn],next[maxn<<1],to[maxn<<1],c[maxn<<1],tt;
llg dn[maxn],up[maxn],ans;//dn就是向下走的期望长度,up就是向上走的期望长度
bool vis[maxn]; int getint(){
int w=0;bool q=0;
char c=getchar();
while((c>'9'||c<'0')&&c!='-') c=getchar();
if(c=='-') c=getchar(),q=1;
while(c>='0'&&c<='9') w=w*10+c-'0',c=getchar();
return q?-w:w;
} void link(int x,int y){
to[++tt]=y;next[tt]=head[x];head[x]=tt;
to[++tt]=x;next[tt]=head[y];head[y]=tt;
c[tt-1]=c[tt]=getint();
} void down(int u){//求向下的期望值
int cnt=0; vis[u]=1;
for(int i=head[u],v;v=to[i],i;i=next[i])
if(!vis[v]) dis[v]=c[i],down(v),cnt++,dn[u]+=dn[v]+c[i];
if(cnt) dn[u]/=cnt; du[u]=cnt; vis[u]=0;
} void dfs(int u,int fa){//求向上的期望值
vis[u]=1; if(fa) hf[u]=1;
if(fa>0) up[u]+=(up[fa]*hf[fa]+du[fa]*dn[fa]-dn[u]-dis[u])/(du[fa]-1+hf[fa]);
for(int i=head[u],v;v=to[i],i;i=next[i])
if(!vis[v]) up[v]+=c[i],dfs(v,u);
} void getrt(int rt,int x){//抠环
for(int u=x;u!=fa[rt];u=fa[u])//dis[u]表示u向它的父亲走的距离
a[++la]=u,b[la+1]=dis[u],vis[u]=1,hf[u]=2;
} void Tarjan(int u,int ff){//抠环
dfn[u]=++tt; fa[u]=ff;
for(int i=head[u],v;v=to[i],i;i=next[i])
if(!dfn[v]) dis[v]=c[i],Tarjan(v,u);
else if(dfn[v]>dfn[u]) b[1]=c[i],getrt(u,v);
} void solve(int u,int j,int z){//在环上走
llg now=0.5,di=0;//di即为在环上走过的距离,now为当前的概率
for(int i=1;i<la;i++){
if(z==-1) di+=b[j]; j+=z;
if(j==la+1) j=1; if(j==0) j=la;
if(z==1) di+=b[j];
if(i==la-1){up[u]+=now*(di+dn[a[j]]);}
else up[u]+=now*du[a[j]]*(dn[a[j]]+di)/(du[a[j]]+1);
now/=(du[a[j]]+1);
}
} void work(){//特殊处理环
tt=0; Tarjan(1,0);
for(int i=1;i<=la;i++) down(a[i]),vis[a[i]]=1;//先求出环上每个点向下的期望
for(int i=1,u;u=a[i],i<=la;i++) solve(u,i,1),solve(u,i,-1);//正反两种走法在环上走
for(int i=1;i<=la;i++) dfs(a[i],0);//子树内求解向上走的期望值
} int main(){
File("park");
n=getint(); m=getint();
for(int i=1;i<=m;i++) link(getint(),getint());
if(m==n-1) down(1),dfs(1,0);
else work();
for(int i=1;i<=n;i++)
ans+=(up[i]*hf[i]+dn[i]*du[i])/(du[i]+hf[i]);
printf("%.5lf",ans/n);
return 0;
}

BZOJ 2878 【NOI2012】 迷失游乐园的更多相关文章

  1. BZOJ 2878: [Noi2012]迷失游乐园( 树形dp )

    一棵树的话直接树形dp(求出往下走和往上走的期望长度). 假如是环套树, 环上的每棵树自己做一遍树形dp, 然后暴力枚举(环上的点<=20)环上每个点跑经过环上的路径就OK了. -------- ...

  2. bzoj 2878 [Noi2012]迷失游乐园——树上的期望dp

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2878 很好的树上概率题的思路,就是分成up和down. 代码中有众多小细节.让我弃疗好几天的 ...

  3. bzoj 2878: [Noi2012]迷失游乐园

    #include<iostream> #include<cstring> #include<cstdio> #define M 100005 #define ld ...

  4. bzoj 2878: [Noi2012]迷失游乐园【树上期望dp+基环树】

    参考:https://blog.csdn.net/shiyukun1998/article/details/44684947 先看对于树的情况 设d[u]为点u向儿子走的期望长度和,du[u]为u点的 ...

  5. 【BZOJ 2878】 2878: [Noi2012]迷失游乐园 (环套树、树形概率DP)

    2878: [Noi2012]迷失游乐园 Description 放假了,小Z觉得呆在家里特别无聊,于是决定一个人去游乐园玩.进入游乐园后,小Z看了看游乐园的地图,发现可以将游乐园抽象成有n个景点.m ...

  6. 2878: [Noi2012]迷失游乐园 - BZOJ

    Description 放假了,小Z觉得呆在家里特别无聊,于是决定一个人去游乐园玩.进入游乐园后,小Z看了看游乐园的地图,发现可以将游乐园抽象成有n个景点.m条道路的无向连通图,且该图中至多有一个环( ...

  7. BZOJ 2878([Noi2012]-失落的游乐园树DP+出站年轮加+后市展望DP+vector的erase)

    2878: [Noi2012]迷失乐园 Time Limit: 10 Sec  Memory Limit: 512 MBSec  Special Judge Submit: 319  Solved:  ...

  8. [bzoj2878][Noi2012]迷失游乐园(基环树dp)

    [bzoj2878][Noi2012]迷失游乐园(基环树dp) bzoj luogu 题意:一颗数或是基环树,随机从某个点开始一直走,不走已经到过的点,求无路可走时的路径长期望. 对于一棵树: 用两个 ...

  9. 【BZOJ 2878】 [Noi2012]迷失游乐园

    Description 放假了,小Z觉得呆在家里特别无聊,于是决定一个人去游乐园玩.进入游乐园后,小Z看了看游乐园的地图,发现可以将游乐园抽象成有n个景点.m条道路的无向连通图,且该图中至多有一个环( ...

  10. NOI2012 : 迷失游乐园

    终于补完NOI2012了好开心~ 题目大意:给定一棵树或者环套外向树,求出从中随机选一条简单路径的期望长度,环上点数不超过20. 设 d[x]表示x的度数,ch[x]表示x孩子个数 up[x]表示x向 ...

随机推荐

  1. NO--11关于"this"你知道多少

    为了更好地理解 this,将 this 使用的场景分成三类: 在函数内部 this 一个额外的,通常是隐含的参数. 在函数外部(顶级作用域中): 这指的是浏览器中的全局对象或者 Node.js 中一个 ...

  2. Vuejs 使用 lib 库模式打包 umd 解决 NPM 包发布的问题

    由于升级了 v0.2 版 GearCase 使用打包工具从 parcel 更换成 vue-cli 3.x.因此打包后发布 NPM 包的方式与之前有很大的差异,这也导致了在发布完 GearCase v0 ...

  3. java.io.tmpdir指定的路径在哪?

    Java.io.tmpdir介绍 System.getproperty(“java.io.tmpdir”)是获取操作系统缓存的临时目录,不同操作系统的缓存临时目录不一样, 在Windows的缓存目录为 ...

  4. python编辑选课系统

    一.需求分析 1. 创建北京.上海 2 所学校 2. 创建linux , python , go 3个课程 , linux\py 在北京开, go 在上海开 3. 课程包含,周期,价格,通过学校创建课 ...

  5. Hyperledger Fabric 1.2 --- Chaincode Operator 解读和测试(二)

    本文接上一节是测试部分 搭建一个模拟测试环境 作者将fabric release1.2工程中的 example-e2e进行了改造来进行本次实验: (1)首先我们将examples/e2e_cli/sc ...

  6. MATLAB2018a与2016b分类学习模型Java库向上兼容操作

    matlab2016版本无法兼容matlab2018版本Classification Learner创建的分类器 在java环境下,使用matlab2016版本调用matlab2018版本Classi ...

  7. lspci命令详解

    基础命令学习目录首页 最近经常用到 lspci -nn | grep Eth 命令,需要学习下PCI总线,找到一篇文章,虽然也是转载,但写的较清晰,再次转载下. http://blog.csdn.ne ...

  8. plsql 不修改tnsnames.ora文件

    PLSQL 不修改tnsname直接连数据库的方式在PLSQL的Database中直接输入192.168.1.6:1521/VP.其中192.168.1.6为数据库的IP:1521为数据库端口:VP为 ...

  9. Spring Bean注册解析(二)

           在上文Spring Bean注册解析(一)中,我们讲解了Spring在注册Bean之前进行了哪些前期工作,以及Spring是如何存储注册的Bean的,并且详细介绍了Spring是如何解析 ...

  10. Mongodb For Mac OSX && 登录验证

    题外话:尽管有不少人贴出了 <我不用mongodb的十大理由> 等系列文章,但是 NoSQL 的发展不会因此而止步, mongodb 是 NoSQL 的典型代表,楼主还是抱乐观态度的,有人 ...