Description

Morenan被困在了一个迷宫里。迷宫可以视为N个点M条边的有向图,其中Morenan处于起点S,迷宫的终点设为T。可惜的是,Morenan非常的脑小,他只会从一个点出发随机沿着一条从该点出发的有向边,到达另一个点。这样,Morenan走的步数可能很长,也可能是无限,更可能到不了终点。若到不了终点,则步数视为无穷大。但你必须想方设法求出Morenan所走步数的期望值。

Input

第1行4个整数,N,M,S,T
第[2, M+1]行每行两个整数o1, o2,表示有一条从o1到o2的边。

Output

一个浮点数,保留小数点3位,为步数的期望值。若期望值为无穷大,则输出"INF"。
【样例输入1】
6 6 1 6
1 2
1 3
2 4
3 5
4 6
5 6
【样例输出1】
3.000
【样例输入2】
9 12 1 9
1 2
2 3
3 1
3 4
3 7
4 5
5 6
6 4
6 7
7 8
8 9
9 7
【样例输出2】
9.500
【样例输入3】
2 0 1 2
【样例输出3】
INF
【数据范围】
测试点
N
M
Hint
[1, 6]
<=10
<=100
 
[7, 12]
<=200
<=10000
 
[13, 20]
<=10000
<=1000000
保证强连通分量的大小不超过100
 
 
另外,均匀分布着40%的数据,图中没有环,也没有自环

【思路】

节点u的期望为E(u)=ΣE(v)/deg(u)+1,移项变成deg(u)*E(u)-ΣE(v)=deg(u),而且题目中说到每一个scc中的点数不超过100。我们就可以先划一下scc,将多个节点方程联立,在每个scc中用高斯消元求每个结点的E,O(n^2)。然后再缩点后的图上dfs统计一下就可以啦。

需要注意应该把t的所有出边切掉,因为我们规定E(t)=0。还有就是如果一个scc中有连往scc外的边,我们把常数项加个E

因为我们切了几条边,所以判断得改一下,不能直接判断sccno[S]是否直接到达sccno[T],可以两个点都dfs一遍,如果有sccno[S]不可以到达但sccno[T]可以到达的则为INF,此时出现有一个scc不能到达sccno[T]的情况,于是可以在这个scc中转来转去,即inf。

【代码】

 #include<stack>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; const int N = 1e4+; vector<int> g[][N];
int n,m,S,T,vis[N],deg[N];
double ans[N],mat[][]; int pre[N],sccno[N],id[N],lowlink[N],dfsc,scccnt;
stack<int> st; vector<int> scc[N]; void tarjan(int flag,int u) {
pre[u]=lowlink[u]=++dfsc;
st.push(u); deg[u]=(int)g[flag][u].size();
for(int i=;i<deg[u];i++) {
int v=g[flag][u][i];
if(!pre[v]) {
tarjan(flag,v);
lowlink[u]=min(lowlink[u],lowlink[v]);
}
else if(!sccno[v])
lowlink[u]=min(lowlink[u],pre[v]);
}
if(pre[u]==lowlink[u]) {
++scccnt;
for(;;) {
int x=st.top(); st.pop();
sccno[x]=scccnt;
scc[scccnt].push_back(x);
id[x]=(int)scc[scccnt].size()-;
if(x==u) break;
}
}
} void gause(int x) {
int n=(int)scc[x].size(),i,j,k,r;
for(i=;i<n;i++) {
int u=scc[x][i];
for(j=;j<n;j++) mat[i][j]=;
mat[i][n]=deg[u];
for(j=;j<g[][u].size();j++) {
int v=g[][u][j];
if(sccno[v]==x) {
mat[i][id[v]]--;
} else {
mat[i][n]+=ans[v];
}
}
mat[i][i]+=deg[u];
}
for(i=;i<n;i++) {
r=i;
for(j=i+;j<n;j++)
if(fabs(mat[j][i])>fabs(mat[r][i])) r=j;
if(r!=i) for(j=;j<=n;j++) swap(mat[i][j],mat[r][j]);
for(k=i+;k<n;k++) {
double f=mat[k][i]/mat[i][i];
for(j=i;j<=n;j++) mat[k][j]-=f*mat[i][j];
}
}
for(i=n-;i>=;i--) {
for(j=i+;j<n;j++)
mat[i][n]-=mat[j][n]*mat[i][j];
mat[i][n]/=mat[i][i];
}
for(i=;i<n;i++)
ans[scc[x][i]]=mat[i][n];
} void solve(int u) {
if(u==sccno[T]) {
ans[T]=; return ;
}
for(int i=;i<g[][u].size();i++) {
int v=g[][u][i];
if(!vis[v]) solve(v);
}
vis[u]=;
gause(u);
} int mark[][N];
void dfs(int flag,int u) {
mark[flag][u]=;
for(int i=;i<g[flag][u].size();i++) {
int v=g[flag][u][i];
if(!mark[flag][v]) dfs(flag,v);
}
}
int main() {
scanf("%d%d%d%d",&n,&m,&S,&T);
S--,T--;
int u,v;
for(int i=;i<m;i++) {
scanf("%d%d",&u,&v);
u-- , v--;
if(u==T) continue;
g[][u].push_back(v);
}
for(int i=;i<n;i++)
if(!pre[i]) tarjan(,i);
for(int i=;i<n;i++) {
for(int j=;j<g[][i].size();j++) {
int v=g[][i][j];
if(sccno[i]!=sccno[v]) {
g[][sccno[i]].push_back(sccno[v]);
g[][sccno[v]].push_back(sccno[i]);
}
}
}
dfs(,sccno[S]); dfs(,sccno[T]);
for(int i=;i<=scccnt;i++)
if(mark[][i]&&!mark[][i]) {
puts("INF"); return ;
}
solve(sccno[S]);
printf("%.3f",ans[S]);
return ;
}

PS:

这个题=-= 好神啊0_0

其实第一眼蒟蒻是想dfs来着,但是有圈啊……有圈啊……有圈啊……

bzoj 2707 [SDOI2012]走迷宫(SCC+高斯消元)的更多相关文章

  1. BZOJ 2707: [SDOI2012]走迷宫( tarjan + 高斯消元 )

    数据范围太大不能直接高斯消元, tarjan缩点然后按拓扑逆序对每个强连通分量高斯消元就可以了. E(u) = 1 + Σ E(v) / degree(u) 对拍时发现网上2个程序的INF判断和我不一 ...

  2. BZOJ 2707: [SDOI2012]走迷宫 拓扑+高斯消元+期望概率dp+Tarjan

    先Tarjan缩点 强连通分量里用高斯消元外面直接转移 注意删掉终点出边和拓扑 #include<cstdio> #include<cstring> #include<a ...

  3. 洛谷 P6030 - [SDOI2012]走迷宫(高斯消元+SCC 缩点)

    题面传送门 之所以写个题解是因为题解区大部分题解的做法都有 bug(u1s1 周六上午在讨论区里连发两个 hack 的是我,由于我被禁言才让 ycx 代发的) 首先碰到这种期望题,我们套路地设 \(d ...

  4. BZOJ 2707: [SDOI2012]走迷宫 [高斯消元 scc缩点]

    2707: [SDOI2012]走迷宫 题意:求s走到t期望步数,\(n \le 10^4\),保证\(|SCC| \le 100\) 求scc缩点,每个scc高斯消元,scc之间直接DP 注意每次清 ...

  5. BZOJ.2707.[SDOI2012]走迷宫(期望 Tarjan 高斯消元)

    题目链接 一个点到达终点的期望步数 \(E_i=\sum_{(i,j)\in G}\frac{E_j+1}{out[i]}\),\(out[i]\)为点\(i\)的出度. 那么对于一个DAG可以直接在 ...

  6. bzoj千题计划289:bzoj 2707: [SDOI2012]走迷宫

    http://www.lydsy.com/JudgeOnline/problem.php?id=2707 dp[i] 表示从点i到终点的期望步数 dp[i]= Σ (dp[j]+1)/out[i] j ...

  7. BZOJ 3143 游走(贪心+期望+高斯消元)

    一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这条边的编号的分 ...

  8. BZOJ 3143 游走 | 数学期望 高斯消元

    啊 我永远喜欢期望题 BZOJ 3143 游走 题意 有一个n个点m条边的无向联通图,每条边按1~m编号,从1号点出发,每次随机选择与当前点相连的一条边,走到这条边的另一个端点,一旦走到n号节点就停下 ...

  9. BZOJ.3143.[HNOI2013]游走(概率 期望 高斯消元)

    题目链接 参考 远航之曲 把走每条边的概率乘上分配的标号就是它的期望,所以我们肯定是把大的编号分配给走的概率最低的边. 我们只要计算出经过所有点的概率,就可以得出经过一条边(\(u->v\))的 ...

随机推荐

  1. DataTable数据进行排序、检索、合并、分页、统计

    在做程序时经常遇到要将反复对数据进行筛选.求和.排序.分页等的情况.每次的数据操作都要去访问数据库很明显是不合理的!当然需要实时数据的情况除外,不做讨论哈.今天无意间在网上看到了这篇文章,挺实用的,拿 ...

  2. Centos 6.2上安装使用 Informix11.70 数据库

    环境要求:操作系统: Centos 6.2 32位数据库软件: iif.11.70.UC7IE.Linux-RHEL5.tar(在IBM网站上注册个帐号就可以下载,包括windows,Linux,Un ...

  3. C#中的lock关键字(初识)

    http://kb.cnblogs.com/page/88513/ 首先给出MSDN的定义: lock 关键字可以用来确保代码块完成运行,而不会被其他线程中断.这是通过在代码块运行期间为给定对象获取互 ...

  4. Java 中正确使用 hashCode 和 equals 方法

    在这篇文章中,我将告诉大家我对hashCode和equals方法的理解.我将讨论他们的默认实现,以及如何正确的重写他们.我也将使用Apache Commons提供的工具包做一个实现. 目录: hash ...

  5. iOS8定位问题

    正文:主要解决iOS8以前能定位,但是在iOS8时候无法定位的问题 在iOS8以前,我们的GPS定位是在用户设置的里面显示的是总是使用,但是在iOS8以后,苹果修改了这部分授权,你需要多加入2个pli ...

  6. 关于怎样拆毁Cuttheprice

    最近我的chrome浏览器被一个叫cuttheprice的插件病毒感染了,总是弹出广告,真是让人抓狂,本来最近几天在复习算法,准备面试的,昨天忍了过去,今天实在受不了了,于是就干掉它吧.说干就干,刚开 ...

  7. 计算器(console version)

    题目描述 请用python编写一个计算器的控制台程序,支持加减乘除.乘方.括号.小数点,运算符优先级为括号>乘方>乘除>加减,同级别运算按照从左向右的顺序计算. 输入描述 数字包括& ...

  8. codeforces 388C Fox and Card Game

    刚刚看到这个题感觉是博弈题: 不过有感觉不像,应该是个贪心: 于是就想贪心策略: 举了一个例子: 3 3 1 2 3 4 3 4 1 2 5 4 1 2 5 8 如果他们两个每次都拿对自己最有利的那个 ...

  9. python re.sub

    python re.sub   python re.sub属于python正则的标准库,主要是的功能是用正则匹配要替换的字符串然后把它替换成自己想要的字符串的方法下面给个例子:import relin ...

  10. Echarts-JAVA

    http://www.oschina.net/p/echarts-java http://my.oschina.net/flags/blog/316920