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

bzoj luogu

题意:一颗数或是基环树,随机从某个点开始一直走,不走已经到过的点,求无路可走时的路径长期望。

对于一棵树:

用两个$dp$数组分别记录从这个点起向上向下走的期望

向下走的$dp$不用多说

向上走的$dp$:

对于从$u$计算$v$的dp

$dp[v]$应当是从u向周围引出所有路径减去走向t的路径的期望后再除以$deg_{u}-1$

对于基环树:

环上的点很少。

此时环上的点的向上$dp$指从u出发向环上两头走的期望。

如何计算:对于环上每一个点都向环的两头各dp一次取平均值。

完毕。

#include<cstdio>
#include<algorithm>
using namespace std;
const int N=100011;
const double eps=1e-8;
template<typename TP>void read(TP &kk){
#define ak *
TP phy=0,ioi=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')ioi=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){phy=phy*10+ch-'0';ch=getchar();}
kk=phy ak ioi;
}
int n,m;
struct sumireko
{
int to,ne,w;
}e[N*2];
int he[N],ecnt;
void addline(int f,int t,int w)
{
e[++ecnt].to=t;
e[ecnt].ne=he[f];
he[f]=ecnt;
e[ecnt].w=w;
}
bool onr[N];
double deg[N],sc[N],dpu[N],dpd[N];
void dfs1(int x,int f)
{
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(t==f||onr[t]) continue;
dfs1(t,x);
dpd[x]+=dpd[t]+e[i].w;
}
if(sc[x]>eps) dpd[x]/=sc[x];
}
void dfs2(int x,int f)
{
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(f==t||onr[t]) continue;
dpu[t]=e[i].w;
if(deg[x]-1.0>eps)dpu[t]+=(dpd[x]*sc[x]-e[i].w-dpd[t]+dpu[x]*(deg[x]-sc[x]))/(deg[x]-1.0);
dfs2(t,x);
}
}
int rnd[N],hp;
int sp;bool vv[N];
int find(int x,int f)
{
if(vv[x]){sp=x;return 1;}
vv[x]=1;int tmp;
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(t==f) continue;
if(tmp=find(t,x))
{
if(tmp==1)
{
rnd[++hp]=x;
onr[x]=1;
if(x!=sp) return 1;
}
return 2;
}
}
return 0;
}
double dpt[N];
void dfs3(int x,int f,int o,int s)
{
if(!o)
{
int g=0;
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(!onr[t]) continue;
g++;dfs3(t,x,g,s);
dpu[x]+=dpt[t]+e[i].w;
}
dpu[x]/=2;
return;
}
double tmp=0;
dpt[x]=0;
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(!onr[t]||t==s||t==f) continue;
dfs3(t,x,o,s);
dpt[x]+=dpt[t]+e[i].w;
tmp+=1;
}
if(sc[x]+tmp>eps)
dpt[x]=(dpt[x]+dpd[x]*sc[x])/(sc[x]+tmp);
} double ans;
int xi,yi,wi;
int main()
{
read(n),read(m);
for(int i=1;i<=m;i++)
{
read(xi),read(yi),read(wi);
addline(xi,yi,wi),addline(yi,xi,wi),deg[xi]+=1,deg[yi]+=1;
}
if(m==n-1)
{
for(int i=1;i<=n;i++) sc[i]=deg[i]-1;
sc[1]+=1;
dfs1(1,0);
dfs2(1,0);
for(int i=1;i<=n;i++) ans+=(dpd[i]*sc[i]+dpu[i])/deg[i];
ans/=n;
printf("%.5lf\n",ans);
return 0;
}
find(1,0);
for(int i=1;i<=n;i++) sc[i]=deg[i]-(onr[i]?2.0:1.0);
for(int i=1;i<=hp;i++) dfs1(rnd[i],0);
for(int i=1;i<=hp;i++) dfs3(rnd[i],0,0,rnd[i]);
for(int i=1;i<=hp;i++) dfs2(rnd[i],0);
for(int i=1;i<=n;i++) ans+=(dpd[i]*sc[i]+dpu[i]*(deg[i]-sc[i]))/deg[i];
ans/=n;
printf("%.5lf\n",ans);
return 0;
}

[bzoj2878][Noi2012]迷失游乐园(基环树dp)的更多相关文章

  1. [BZOJ2878][NOI2012]迷失游乐园(环套树DP+概率)

    推荐讲解:https://www.cnblogs.com/Tunix/p/4561493.html 首先考虑树的情况,就是经典的树上概率DP.先DP出down表示从这个点向儿子走能走的期望长度,再DP ...

  2. BZOJ2878 NOI2012迷失游乐园(树形dp+环套树+概率期望)

    考虑树的部分分怎么做.令f[i]为i向子树内走的期望路径长度,转移比较显然.算答案时先把其父亲的答案弄好就可以统计自己的答案了. 环套树也类似.树里直接dp,对环上点暴力考虑环上的每条路径,算完后再在 ...

  3. bzoj2878 [Noi2012]迷失游乐园——概率期望DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2878 这个博客写得很好:https://www.cnblogs.com/qt666/p/72 ...

  4. BZOJ2878 [Noi2012]迷失游乐园 【基环树 + 树形dp + 期望dp】

    题目链接 BZOJ2878 题解 除了实现起来比较长,思维难度还是挺小的 观察数据范围发现环长不超过\(20\),而我们去掉环上任何一个点就可以形成森林 于是乎我们枚举断掉的点,然后只需求出剩余每个点 ...

  5. [luogu2081 NOI2012] 迷失游乐园 (树形期望dp 基环树)

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

  6. bzoj2878 [Noi2012]迷失游乐园 [树形dp]

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

  7. BZOJ2878 [Noi2012]迷失游乐园

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...

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

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

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

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

随机推荐

  1. Solution -「洛谷 P5325」Min_25 筛

    \(\mathcal{Description}\)   Link.   对于积性函数 \(f(x)\),有 \(f(p^k)=p^k(p^k-1)~(p\in\mathbb P,k\in\mathbb ...

  2. 第三章 in,not in 的理解,limit限制结果集

     IN 运算符用来判断表达式的值是否位于给出的列表中:如果是,返回值为 1,否则返回值为 0.NOT IN 的作用和 IN 恰好相反,NOT IN 用来判断表达式的值是否不存在于给出的列表中:如果不是 ...

  3. python2写ping监控,自动发现ip

    玩了hostmonitor,老外写的很好.但是不符合国情,只有邮件适合发送. 今天用python 写一个自动发现ip,ping失败报警的程序.(微信和邮件报警) 以前用python写的发微信,发邮件直 ...

  4. 《手把手教你》系列技巧篇(六十九)-java+ selenium自动化测试 - 读取csv文件(详细教程)

    1.简介 在实际测试中,我们不仅需要读取Excle,而且有时候还需要读取CSV类的文件.如何去读取CSV的文件,宏哥今天就讲解和分享一下,希望对你能够有所帮助.前面介绍了如何读取excel文件,本篇介 ...

  5. Nginx频繁报“500 Internal Server Error”错误

    服务器导致访问量激增,频繁报"500 Internal Server Error"错误.我查了一下nginx的错误日志(apt-get方式安装的nginx的错误日志在/var/lo ...

  6. 【Kotlin】初识Kotlin之面向对象

    [Kotlin]初识Kotlin之面向对象 1.类 在Kotlin中,类用关键字class来定义 如果一个类具有类体,那么需要使用{ }来写类体内容,如果不需要类体,那么只需要定义类名就可以了 // ...

  7. BI系统:发挥大数据的价值

    ​大数据是指大数据集,这些数据集经过计算分析以揭示与数据的某个方面相关的模式和趋势.首先,还是要重新审视大数据的定义.行业里对大数据的定义有很多,有广义的定义,也有狭义的定义. 大数据的分析与挖掘,把 ...

  8. c# 编程学习(四)

    逻辑AND(逻辑与)操作符(用&&表示)和逻辑OR(逻辑或) 操作符(用||表示).这两个操作符统称条件逻辑操作符.只有作为操作数的两个布尔表达式都为 true.操作符&&am ...

  9. 针对python输入要求

    针对python输入要求 类型: 1.输入行数不确定,并且每一行输入一个数据. a=[] b=input() while b!='-1': //指随意使用一个值作为一个标志,来进行控制输入的行数.(在 ...

  10. Qt:Qt资源系统

    学习自 Qt 资源系统(Qt Resource System) - 知乎 1.什么是Qt 资源系统 Qt资源系统是一种将图片.数据存储于二进制文件中的一套系统.这些图片.数据会被我们的程序使用,它们称 ...