Portal --> hdu4035

Solution

  讲道理不是很懂为啥概d那么喜欢走迷宫qwq

  (推式子推的很爽的一题?)

  首先大力dp列式子

  用\(f[i]\)表示从\(i\)到离开的期望步数,\(du[i]\)表示\(i\)点的度数,那么可以得到:

\[f[i]=e_i* 0+k_i* f[1]+\frac{1-e_i-k_i}{du[i]}\sum(f[u]+1)
\]

  其中\(u\)是\(i\)的儿子

  稍微整理一下,对于叶子节点和普通节点我们可以分别得到这样的两条式子:

\[\begin{aligned}
&叶子:f[i]=k_i*f[1]+(1-e_i-k_i)*f[fa(i)]+(1-e_i-k_i)\\
\\
&普通:f[i]=k_i*f[1]+\frac{1-e_i-k_i}{du[i]}*f[fa(i)]+\frac{1-e_i-k_i}{du[i]}*\sum f[u]+(1-e_i-k_i)\\
\end{aligned}
\]

  考虑从下往上推,那么将\(f[u]\)看成常数的话,其实我们可以将\(f[i]\)表达为\(A_i*f[1]+B_i*f[fa(i)]+C_i\)这样的形式

  

  这个时候看一下我们要求的答案,可以表示成:\(f[1]=A_1*f[1]+B_1*0+C_1\)

  那也就是说,\(f[1]=\frac{C_1}{1-A_1}\)

  那。。所以其实到最后根本不需要求出所有的\(f\)值只要知道系数我们就可以直接把答案求出来了(大快人心事!)

  那剩下的就是要想办法把系数求出来

  

  化简一下上面的\(f\)的表达式

  我们发现\(\sum f[u]\)这个东西其实可以写成一个与\(f[i]\)相关的表达式(因为\(fa(u)=i\)),也就是:

\[\begin{aligned}
\sum f[u]&=\sum A_u*f[1]+B_u*f[i]+C_u
\end{aligned}
\]

  把这个式子带到上面普通节点的\(f\)值的表达式里面去,把跟\(f[i]\)有关的放一边,其他放另一边:

\[\begin{aligned}
f[i]&=k_i*f[1]+\frac{1-e_i-k_i}{du[i]}*f[fa(i)]+\frac{1-e_i-k_i}{du[i]}*(f[1]*\sum A_u+f[i]*\sum B_u+\sum C_u)\\
\\
(1-\frac{1-e_i-k_i}{du[i]}*\sum B_u)*f[i]&=(k_i+\frac{1-e_i-k_i}{du[i]}*\sum A_u)*f[1]+\frac{1-e_i-k_i}{du[i]}*f[fa(i)]+\frac{1-e_i-k_i}{du[i]}*(\sum C_u+du[i])
\end{aligned}
\]

  

  然后我们就可以得出\(、、A、B、C\)这三个系数的表达式啦

\[\begin{aligned}
叶子:&A_i=k_i\\
&B_i=1-e_i-k_i\\
&C_i=1-e_i-k_i\\
\\
普通:&A_i=(k_i+\frac{1-e_i-k_i}{du[i]}*\sum A_u)/(1-\frac{1-e_i-k_i}{du[i]}*\sum B_u)\\
&B_i=\frac{1-e_i-k_i}{du[i]}/(1-\frac{1-e_i-k_i}{du[i]}*\sum B_u)\\
&C_i=\frac{1-e_i-k_i}{du[i]}*(\sum C_u+du[i])/(1-\frac{1-e_i-k_i}{du[i]}*\sum B_u)\\
\end{aligned}
\]

  看上去长得。。挺丑。。但是!

  得出了这个式子之后我们就可以从下往上将所有的系数求出来了

​  

  然后就做完了

  那么无解是什么情况呢?应该就是在中间计算过程中,如果有哪一步算的时候除以\(0\)了就是无解了,中间算的时候判断一下就好了

  

  代码大概长这个样子

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
const int MAXN=1e4+10;
const double eps=1e-9;
struct xxx{
int y,nxt;
}a[MAXN*2];
int h[MAXN],du[MAXN];
double A[MAXN],B[MAXN],C[MAXN],k[MAXN],e[MAXN];
int n,m,tot,T;
void add(int x,int y);
bool dfs(int fa,int x);
void init(); int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
int x,y;
double ans;
scanf("%d",&T);
for (int o=1;o<=T;++o){
printf("Case %d: ",o);
scanf("%d",&n);
init();
for (int i=1;i<n;++i){
scanf("%d%d",&x,&y);
add(x,y); add(y,x);
++du[x]; ++du[y];
}
for (int i=1;i<=n;++i){
scanf("%d%d\n",&x,&y);
k[i]=1.0*x/100.0;
e[i]=1.0*y/100.0;
}
if (dfs(0,1)){
if (fabs(1-A[1])<=eps) printf("impossible\n");
else
printf("%.6lf\n",C[1]/(1.0-A[1]));
}
else
printf("impossible\n");
}
} void init(){
tot=0;
memset(h,-1,sizeof(h));
memset(du,0,sizeof(du));
} void add(int x,int y){
a[++tot].y=y; a[tot].nxt=h[x]; h[x]=tot;
} bool dfs(int fa,int x){
int u,son=0;
double sum=0;
A[x]=k[x]; B[x]=(1.0-e[x]-k[x])/(1.0*du[x]); C[x]=du[x];
for (int i=h[x];i!=-1;i=a[i].nxt){
u=a[i].y;
if (u==fa) continue;
++son;
if (!dfs(x,u)) return false;
C[x]+=C[u];
A[x]+=(1.0-e[x]-k[x])/(1.0*du[x])*A[u];
sum+=B[u];
}
if (son==0){
A[x]=k[x]; B[x]=1.0-e[x]-k[x]; C[x]=1.0-k[x]-e[x];
}
else{
C[x]*=(1-e[x]-k[x])/(1.0*du[x]);
sum*=(1.0-e[x]-k[x])/(1.0*du[x]);
sum=1.0-sum;
if (fabs(sum)<eps) return false;
A[x]/=sum;
B[x]/=sum;
C[x]/=sum;
}
return true;
}

【hdu4035】Maze的更多相关文章

  1. 【CF123E】Maze

    Portal --> cf123E Solution 首先步数的话可以转化成每条边经过了几次这样来算 假设现在确定了起点\(S\)和终点\(T\),我们将\(T\)看成树根,那么考虑边\((u, ...

  2. 【XSY2525】Maze 2017多校

    Description 考虑一个 N×M 的网格,每个网格要么是空的,要么是障碍物.整个网格四周都是墙壁(即第1行和第n行,第1列和第m列都是墙壁),墙壁有且仅有两处开口,分别代表起点和终点.起点总是 ...

  3. 【南京邮电】maze 迷宫解法

    [南京邮电]maze 迷宫解法 题目来源:南京邮电大学网络攻防训练平台. 题目下载地址:https://pan.baidu.com/s/1i5gLzIt (密码rijss) 0x0 初步分析 题目中给 ...

  4. 【】maze

    [链接]点击打开链接 [题意] 小 T 被放到了一个迷宫之中,这个迷宫由 n 个节点构成,两个节点之间可能存在多条无 向边,小 T 的起点为 1 号节点,终点为 n 号节点.有 m 条无向边,对于每一 ...

  5. 【转载】图论 500题——主要为hdu/poj/zoj

    转自——http://blog.csdn.net/qwe20060514/article/details/8112550 =============================以下是最小生成树+并 ...

  6. 【bfs】迷宫问题

    [题目描述] 定义一个二维数组: int maze[5][5] = { 0,1,0,0,0, 0,1,0,1,0, 0,0,0,0,0, 0,1,1,1,0, 0,0,0,1,0, }; 它表示一个迷 ...

  7. 【HDOJ图论题集】【转】

    =============================以下是最小生成树+并查集====================================== [HDU] How Many Table ...

  8. 【LeetCode】深搜DFS(共85题)

    [98]Validate Binary Search Tree [99]Recover Binary Search Tree [100]Same Tree [101]Symmetric Tree [1 ...

  9. 【LeetCode】BFS(共43题)

    [101]Symmetric Tree 判断一棵树是不是对称. 题解:直接递归判断了,感觉和bfs没有什么强联系,当然如果你一定要用queue改写的话,勉强也能算bfs. // 这个题目的重点是 比较 ...

随机推荐

  1. animation和transition

    相同点 指定要侦听更改的CSS属性. 设置计时(缓和)功能以改变从一个属性值到另一个属性值的速率 指定持续时间以控制动画或转换所需的时间 以编程方式收听您可以随意执行的动画和特定于转换的事件 可视化C ...

  2. js页面跳转,url带url参数解决方案

    今天,在做一个项目的时候,向后端发送了一个Ajax请求,后端返回了一个字符串,告诉我未登录.那么我需要跳转到登录页面,同时告诉登录页面,登录成功后,需要跳回的url.也就是标题所说,url中的一个参数 ...

  3. Bin Packing 装箱问题——NPH问题的暴力枚举 状压DP

    题目: 给定n(1≤n≤24)个物品,重量分别为wi,装进一些容量为S(S<1e8)的背包,最少需要多少个背包?

  4. 第二次ScrumMeeting博客

    第二次ScrumMeeting博客 本次会议于10月26日(四)22时整在3公寓725房间召开,持续15分钟. 与会人员:刘畅.方科栋.辛德泰.窦鑫泽.张安澜. 1. 每个人的工作(有Issue的内容 ...

  5. 经验之谈:10位顶级PHP大师的开发原则

    导读:在Web开发世界里,PHP是最流行的语言之一,从PHP里,你能够很容易的找到你所需的脚本,遗憾的是,很少人会去用“最佳做法”去写一个PHP程序.这里,我们向大家介绍PHP的10种最佳实践,当然, ...

  6. Android:有关下拉菜单导航的学习(供自己参考)

    Android:有关==下拉菜单导航==的学习 因为先前的学习都没想着记录自己的学习历程,所以该博客才那么迟才开始写. 内容: ==下拉菜单导航== 学习网站:android Spinner控件详解 ...

  7. c# dictionnary根据value查找对应的key

    属性方法中并没有包含此功能,因此需要自己自定义一个方法: string regionName = ""; if (ControlForm.swichLanguage.Contain ...

  8. Alpha冲刺总结报告

    一.项目预期计划 允许粗糙的美工设计.由于是毫无经验的人生第一次,必定在开发过程中会遇到许多的问题,因而我们必定会花费不少时间在学习和debug上. 实现除了他山之石和规则系统以外的所有内容. 在日历 ...

  9. 学习c++ofstream和ifstream

    定义数据流对象指针 对文件进行读写操作首先必须要定义一个数据流对象指针,数据流对象指针有三种类型,它们分别是: Ifstream:表示读取文件流,使用的时候必须包含头文件"ifstream& ...

  10. Alpha事后诸葛(团队)

    [设想和目标] Q1:我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? "小葵日记"是为了解决18-30岁年轻用户在记录生活时希望得到一美体验友好 ...