dp求期望的题。
设 E[i]表示在结点i处,要走出迷宫所要走的边数的期望。E[1]即为所求。
叶子结点:
E[i] = ki*E[1] + ei*0 + (1-ki-ei)*(E[father[i]] + 1);
= ki*E[1] + (1-ki-ei)*E[father[i]] + (1-ki-ei); 非叶子结点:(m为与结点相连的边数)
E[i] = ki*E[1] + ei*0 + (1-ki-ei)/m*( E[father[i]]+1 + ∑( E[child[i]]+1 ) );
= ki*E[1] + (1-ki-ei)/m*E[father[i]] + (1-ki-ei)/m*∑(E[child[i]]) + (1-ki-ei); 设对每个结点:E[i] = Ai*E[1] + Bi*E[father[i]] + Ci; 对于非叶子结点i,设j为i的孩子结点,则
∑(E[child[i]]) = ∑E[j]
= ∑(Aj*E[1] + Bj*E[father[j]] + Cj)
= ∑(Aj*E[1] + Bj*E[i] + Cj)
带入上面的式子得
(1 - (1-ki-ei)/m*∑Bj)*E[i] = (ki+(1-ki-ei)/m*∑Aj)*E[1] + (1-ki-ei)/m*E[father[i]] + (1-ki-ei) + (1-ki-ei)/m*∑Cj;
由此可得
Ai = (ki+(1-ki-ei)/m*∑Aj) / (1 - (1-ki-ei)/m*∑Bj);
Bi = (1-ki-ei)/m / (1 - (1-ki-ei)/m*∑Bj);
Ci = ( (1-ki-ei)+(1-ki-ei)/m*∑Cj ) / (1 - (1-ki-ei)/m*∑Bj); 对于叶子结点
Ai = ki;
Bi = 1 - ki - ei;
Ci = 1 - ki - ei; 从叶子结点开始,直到算出 A1,B1,C1; E[1] = A1*E[1] + B1*0 + C1;
所以
E[1] = C1 / (1 - A1);
若 A1趋近于1则无解...

经典DP期望了。以上是题解了,其实这道题是最开始做的,所以后来才会用那种设系数的方法。一直留到现在才写,自己重新推了一遍,感觉这种设系数然后递推系数的方法实在妙极啊。。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define N 10010 struct ed{
int u,v;
int next;
}edge[N*2];
int head[N]; struct nd{
double k,e;
}node[N];
int tot,n;
double A[N],B[N],C[N]; void addedge(int u,int v){
edge[tot].u=u;
edge[tot].v=v;
edge[tot].next=head[u];
head[u]=tot++;
} void dfs(int parent,int now){
bool leaf=true;
double tcA,tcB,tcC;
tcA=tcB=tcC=0;
int cnt=0;
double ki=node[now].k;
double ei=node[now].e;
for(int e=head[now];e!=-1;e=edge[e].next){
cnt++;
if(edge[e].v!=parent){
leaf=false;
dfs(now,edge[e].v);
tcA+=A[edge[e].v];
tcB+=B[edge[e].v];
tcC+=C[edge[e].v];
}
}
if(leaf){
A[now]=ki;
B[now]=1-ki-ei;
C[now]=1-ki-ei;
}
else{
A[now]=(ki+tcA*(1-ki-ei)/cnt)/(1-tcB*(1-ki-ei)/cnt);
B[now]=(1-ki-ei)/cnt/(1-tcB*(1-ki-ei)/cnt);
C[now]=(tcC*(1-ki-ei)/cnt+(1-ki-ei))/(1-tcB*(1-ki-ei)/cnt);
}
} int main(){
int T,u,v,kase=0;
double k,e;
scanf("%d",&T);
while(T--){
scanf("%d",&n);
for(int i=1;i<=n;i++){
head[i]=-1;
A[i]=B[i]=C[i]=0;
}
tot=0;
for(int i=1;i<n;i++){
scanf("%d%d",&u,&v);
addedge(u,v);
addedge(v,u);
}
for(int i=1;i<=n;i++){
scanf("%lf%lf",&k,&e);
node[i].k=k/100;
node[i].e=e/100;
}
dfs(0,1);
double ans=(C[1])/(1-A[1]);
if(fabs(1-A[1])<1e-9) //必须是-9。。。跪了。。
printf("Case %d: impossible\n",++kase);
else{
printf("Case %d: %.6lf\n",++kase,ans);
}
}
return 0;
}

  

最新文章

  1. UIImage
  2. Lodop6 以上打印控件使用,详参考自带说明文档,打印样式及文字大小要特殊设置一下
  3. php 获取代码执行时间和消耗的内存
  4. 批量插入数据 C# SqlBulkCopy使用
  5. vitamio videoView 用隐藏除videoview的控件,并旋转屏幕方向实现的全屏功能,出现的画面不能填充满videoview(画面不完整)
  6. Makefile Shell 脚本;sed命令
  7. 一个有趣的Ajax Hack示范
  8. iis认证方式
  9. OWASP Top 10十大风险 – 10个最重大的Web应用风险与攻防
  10. Javascript定义变量
  11. Saiku部分函数解析(八)
  12. 20135202闫佳歆--week6 进程的描述与创建--学习笔记
  13. vmware产品框架-计算中心,5.1更新等
  14. calico集成详解
  15. codeforces9D How many trees?
  16. Mirror--程序访问镜像数据库的超时机制
  17. Matlab中的数据预处理-归一化(mapminmax)与标准化(mapstd)
  18. tmux分屏
  19. SQLAlchemy技术文档(中文版)-下
  20. python基础小练习

热门文章

  1. JAVA实现将GeoHash转化为相应的经纬度坐标
  2. Python+Django+SAE系列教程16-----cookie&amp;amp;session
  3. HDU 1160 FatMouse&amp;#39;s Speed DP题解
  4. java连接mysql数据库增删改查操作记录
  5. nyoj--1185--最大最小值(线段树)
  6. shp系列(一)——利用C++进行shp文件的读(打开)与写(创建)开言
  7. Elasticsearch部署异常Permission denied
  8. SQL 导出数据字典
  9. vue-cli搭建项目结构及引用bootstrap
  10. 操作mysql操作数据库