【BZOJ2878】【NOI2012】迷失游乐园(动态规划)

题面

BZOJ

题解

记得以前考试的时候做过这道题目

这题的暴力还是非常显然的,每次\(dfs\)一下就好了。

时间复杂度\(O(n^2)\)

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstdlib>
  4. #include<cstring>
  5. #include<cmath>
  6. #include<algorithm>
  7. #include<set>
  8. #include<map>
  9. #include<vector>
  10. #include<queue>
  11. using namespace std;
  12. #define ll long long
  13. #define RG register
  14. #define MAX 111111
  15. inline int read()
  16. {
  17. RG int x=0,t=1;RG char ch=getchar();
  18. while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
  19. if(ch=='-')t=-1,ch=getchar();
  20. while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
  21. return x*t;
  22. }
  23. struct Line{int v,next,w;}e[MAX<<1];
  24. int h[MAX],cnt=1;
  25. inline void Add(int u,int v,int w){e[cnt]=(Line){v,h[u],w};h[u]=cnt++;}
  26. double ans;
  27. bool vis[MAX];
  28. int n,m;
  29. void dfs(int u,double p,int len)
  30. {
  31. int tot=0;vis[u]=true;
  32. for(int i=h[u];i;i=e[i].next)
  33. if(!vis[e[i].v])++tot;
  34. for(int i=h[u];i;i=e[i].next)
  35. if(!vis[e[i].v])
  36. dfs(e[i].v,p/tot,len+e[i].w);
  37. vis[u]=false;
  38. if(!tot)ans+=p*len;
  39. }
  40. int main()
  41. {
  42. n=read();m=read();
  43. for(int i=1;i<=m;++i)
  44. {
  45. int u=read(),v=read(),w=read();
  46. Add(u,v,w);Add(v,u,w);
  47. }
  48. for(int i=1;i<=n;++i)dfs(i,1.0/n,0);
  49. printf("%.5lf\n",ans);
  50. return 0;
  51. }

发现到有一棵树的部分数据点

考虑一下树的答案

显然是以当前点为根节点,到达它所有叶子的路径长度的期望

显然可以树型\(dp\)+换根解决,复杂度\(O(n)\)

综合暴力可以拿到\(80\)分

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstdlib>
  4. #include<cstring>
  5. #include<cmath>
  6. #include<algorithm>
  7. #include<set>
  8. #include<map>
  9. #include<vector>
  10. #include<queue>
  11. using namespace std;
  12. #define ll long long
  13. #define RG register
  14. #define MAX 111111
  15. inline int read()
  16. {
  17. RG int x=0,t=1;RG char ch=getchar();
  18. while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
  19. if(ch=='-')t=-1,ch=getchar();
  20. while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
  21. return x*t;
  22. }
  23. struct Line{int v,next,w;}e[MAX<<1];
  24. int h[MAX],cnt=1;
  25. inline void Add(int u,int v,int w){e[cnt]=(Line){v,h[u],w};h[u]=cnt++;}
  26. double ans;
  27. bool vis[MAX];
  28. int n,m;
  29. void dfs(int u,double p,int len)
  30. {
  31. int tot=0;vis[u]=true;
  32. for(int i=h[u];i;i=e[i].next)
  33. if(!vis[e[i].v])++tot;
  34. for(int i=h[u];i;i=e[i].next)
  35. if(!vis[e[i].v])
  36. dfs(e[i].v,p/tot,len+e[i].w);
  37. vis[u]=false;
  38. if(!tot)ans+=p*len;
  39. }
  40. namespace Tree
  41. {
  42. int son[MAX];
  43. double E[MAX],ans,E2[MAX];
  44. void dfs(int u,int ff)
  45. {
  46. for(int i=h[u];i;i=e[i].next)
  47. {
  48. int v=e[i].v;if(v==ff)continue;
  49. ++son[u];dfs(v,u);E[u]+=E[v]+e[i].w;
  50. }
  51. if(son[u])E[u]/=son[u];
  52. }
  53. void DFS(int u,int ff,int w)
  54. {
  55. if(u==1)E2[u]=E[u];
  56. else
  57. {
  58. E2[u]=E[u]*son[u];
  59. if(son[ff]>1)E2[u]+=(E2[ff]*son[ff]-E[u]-w)/(son[ff]-1);
  60. E2[u]+=w;
  61. E2[u]/=(son[u]+1);
  62. ++son[u];
  63. }
  64. ans+=E2[u];
  65. for(int i=h[u];i;i=e[i].next)
  66. if(e[i].v!=ff)DFS(e[i].v,u,e[i].w);
  67. }
  68. void Solve()
  69. {
  70. dfs(1,0);DFS(1,0,0);ans/=n;
  71. printf("%.5lf\n",ans);
  72. }
  73. }
  74. int main()
  75. {
  76. n=read();m=read();
  77. for(int i=1;i<=m;++i)
  78. {
  79. int u=read(),v=read(),w=read();
  80. Add(u,v,w);Add(v,u,w);
  81. }
  82. if(m==n-1){Tree::Solve();return 0;}
  83. for(int i=1;i<=n;++i)dfs(i,1.0/n,0);
  84. printf("%.5lf\n",ans);
  85. return 0;
  86. }

剩下的问题是如何解决\(n=m\),也就是基环树的问题

我们这样考虑。

首先把环给拉出来,拉直,然后一条边从头连到尾

那么,这样子就是一个环,然后每个点上面挂着一些点

我们显然可以计算出每个点向下的期望,现在要算的是向上的期望

因为向上的期望只可能在环上走,所以枚举所有环上的点,依次考虑每个点的期望

因为在环上只有三种走法,向左,向右,走向子树

因此,对于每个环上的点,暴力\(dfs\)一遍,计算它到达长度的期望,

这个长度显然是到达某个环上的点之后,进入了这个点的子树。

这样子再像树型\(dp\)一样从上往下转移一次就好了。

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstdlib>
  4. #include<cstring>
  5. #include<cmath>
  6. #include<algorithm>
  7. #include<set>
  8. #include<map>
  9. #include<vector>
  10. #include<queue>
  11. using namespace std;
  12. #define ll long long
  13. #define RG register
  14. #define MAX 111111
  15. inline int read()
  16. {
  17. RG int x=0,t=1;RG char ch=getchar();
  18. while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
  19. if(ch=='-')t=-1,ch=getchar();
  20. while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
  21. return x*t;
  22. }
  23. struct Line{int v,next,w;}e[MAX<<1];
  24. int h[MAX],cnt=1;
  25. inline void Add(int u,int v,int w){e[cnt]=(Line){v,h[u],w};h[u]=cnt++;}
  26. int n,m,son[MAX],root;
  27. double E[MAX],ans,E2[MAX];
  28. bool incir[MAX],vis[MAX];
  29. int fa[MAX],cir[MAX],tot;
  30. void dfs(int u)
  31. {
  32. vis[u]=true;
  33. for(int i=h[u];i;i=e[i].next)
  34. {
  35. int v=e[i].v;if(vis[v]||incir[v])continue;
  36. ++son[u];dfs(v);E[u]+=E2[v]+e[i].w;
  37. }
  38. if(son[u])E2[u]=E[u]/son[u];
  39. if(u!=root)++son[u];
  40. }
  41. void DFS(int u)
  42. {
  43. vis[u]=true;
  44. for(int i=h[u];i;i=e[i].next)
  45. {
  46. int v=e[i].v;if(vis[v]||incir[v])continue;
  47. E[v]+=(E[u]-E2[v]-e[i].w)/max(1,son[u]-1)+e[i].w;
  48. DFS(v);
  49. }
  50. }
  51. void dfscir(int u,int ff)
  52. {
  53. vis[u]=true;fa[u]=ff;
  54. for(int i=h[u];i;i=e[i].next)
  55. {
  56. int v=e[i].v;if(v==ff)continue;
  57. if(incir[v])continue;
  58. if(vis[v])//Circle
  59. {
  60. for(int j=u;j!=v;j=fa[j])
  61. cir[++tot]=j;
  62. cir[++tot]=v;
  63. for(int j=1;j<=tot;++j)incir[cir[j]]=true;
  64. }
  65. else dfscir(v,u);
  66. }
  67. }
  68. double g[MAX],f[MAX];
  69. void dfs(int u,int ff)
  70. {
  71. bool fl=false;g[u]=0;
  72. for(int i=h[u];i;i=e[i].next)
  73. {
  74. int v=e[i].v;if(v==root||v==ff||!incir[v])continue;
  75. fl=true;dfs(v,u);
  76. g[u]+=g[v]+e[i].w;
  77. }
  78. if(u==root)return;
  79. int k=son[u];if(!k)++k;
  80. if(!fl)g[u]=E[u]/k;
  81. else k=son[u]+1,g[u]=(g[u]+E[u])/k;
  82. }
  83. int main()
  84. {
  85. n=read();m=read();
  86. for(int i=1;i<=m;++i)
  87. {
  88. int u=read(),v=read(),w=read();
  89. Add(u,v,w);Add(v,u,w);
  90. }
  91. if(m==n-1){root=1;dfs(1);memset(vis,0,sizeof(vis));DFS(1);}
  92. else
  93. {
  94. dfscir(1,0);memset(vis,0,sizeof(vis));
  95. for(int i=1;i<=tot;++i)root=cir[i],dfs(cir[i]);
  96. for(int i=1;i<=tot;++i)root=cir[i],dfs(cir[i],0),f[cir[i]]=g[cir[i]];
  97. memset(vis,0,sizeof(vis));
  98. for(int i=1;i<=tot;++i)son[cir[i]]+=2,E[cir[i]]+=f[cir[i]];
  99. for(int i=1;i<=tot;++i)root=cir[i],DFS(cir[i]);
  100. }
  101. for(int i=1;i<=n;++i)ans+=E[i]/son[i];ans/=n;
  102. printf("%.5lf\n",ans);
  103. return 0;
  104. }

【BZOJ2878】【NOI2012】迷失游乐园(动态规划)的更多相关文章

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

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

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

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

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

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

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

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

  5. BZOJ2878 [Noi2012]迷失游乐园

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

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

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

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

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

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

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

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

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

  10. 【bzoj2878】 Noi2012—迷失游乐园

    http://www.lydsy.com/JudgeOnline/problem.php?id=2878 (题目链接) 题意 求基环树上以任意点为起点的简单路径期望长度. Solution 啊啊啊好丑 ...

随机推荐

  1. Intellij IDEA 2017 通过scala工程运行wordcount

    首先是安装scala插件,可以通过idea内置的自动安装方式进行,也可以手动下载可用的插件包之后再通过idea导入. scala插件安装完成之后,新建scala项目,右侧使用默认的sbt 点击Next ...

  2. Android开源的精美日历控件,热插拔设计的万能自定义UI

    Android开源的精美日历控件,热插拔设计的万能自定义UI UI框架应该逻辑与界面实现分离,该日历控件使用了热插拔的设计 ,简单几步即可实现你需要的UI效果,热插拔的思想是你提供你的实现,我提供我的 ...

  3. python基础数据类型3

    python_day_5 今日大纲: 1. dict 用大括号{} 括起来. 内部使用key:value的形式来保存数据 {'jay':'周杰伦', "jj":'林俊杰'} 注意: ...

  4. 音频分析框架pyAudioAnalysis文档

    “ pyAudioAnalysis是一个非常好用且强大的音频分析开源工具,能实现音频的特征提取.分类和回归模型的训练和执行,以及其他一些实用的功能.此外,本文档并非直译,也有部分比较简略,可以结合源码 ...

  5. 在jre1.8版本下,使用ikvm将jar转换为dll,以供c#调用

    由于合作方使用.net编程,jar包不能用,需要转换成dll格式,来回转换了十几个dll文件(心塞..),终于生成了一个可用的.在这里将走过的弯弯绕绕总结下,希望遇到相似问题的同好们,能走得顺利些. ...

  6. LearnPython - Zip格式文件的解压缩

    import zipfile import os def unzip(zip_name, target_dir): files = zipfile.ZipFile(zip_name) for zip_ ...

  7. 《数据结构与算法JavaScript描述》中的一处错误

    最近在看<数据结构与算法JavaScript描述>这本书,看到选择排序这部分时,发现一个比较大的错误. 原书的选择排序算法是这样的: function selectionSort() { ...

  8. 细节--服务器mysql空密码

    在部署致服务器的时候 发现mysql密码为空的情况 如果采用 root账户的话 试过很多 比如不写下面这行 <property name="password" value=& ...

  9. NSURLErrorDomain错误代码

    kCFURLErrorUnknown = -998, kCFURLErrorCancelled = -999, kCFURLErrorBadURL = -1000, kCFURLErrorTimedO ...

  10. SSH 框架的心得

    使用SSH框架做完了一个普通网站的前后台项目,成热写点心得,免得以后再入坑.其中使用 Strust2  2.3.33 + Spring 4.3.9 + Hibernate 5.2.10 eclipse ...