bzoj 2707 [SDOI2012]走迷宫(SCC+高斯消元)
Description
Input
Output
|
测试点
|
N
|
M
|
Hint
|
|
[1, 6]
|
<=10
|
<=100
|
|
|
[7, 12]
|
<=200
|
<=10000
|
|
|
[13, 20]
|
<=10000
|
<=1000000
|
保证强连通分量的大小不超过100
|
【思路】
节点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+高斯消元)的更多相关文章
- BZOJ 2707: [SDOI2012]走迷宫( tarjan + 高斯消元 )
数据范围太大不能直接高斯消元, tarjan缩点然后按拓扑逆序对每个强连通分量高斯消元就可以了. E(u) = 1 + Σ E(v) / degree(u) 对拍时发现网上2个程序的INF判断和我不一 ...
- BZOJ 2707: [SDOI2012]走迷宫 拓扑+高斯消元+期望概率dp+Tarjan
先Tarjan缩点 强连通分量里用高斯消元外面直接转移 注意删掉终点出边和拓扑 #include<cstdio> #include<cstring> #include<a ...
- 洛谷 P6030 - [SDOI2012]走迷宫(高斯消元+SCC 缩点)
题面传送门 之所以写个题解是因为题解区大部分题解的做法都有 bug(u1s1 周六上午在讨论区里连发两个 hack 的是我,由于我被禁言才让 ycx 代发的) 首先碰到这种期望题,我们套路地设 \(d ...
- BZOJ 2707: [SDOI2012]走迷宫 [高斯消元 scc缩点]
2707: [SDOI2012]走迷宫 题意:求s走到t期望步数,\(n \le 10^4\),保证\(|SCC| \le 100\) 求scc缩点,每个scc高斯消元,scc之间直接DP 注意每次清 ...
- BZOJ.2707.[SDOI2012]走迷宫(期望 Tarjan 高斯消元)
题目链接 一个点到达终点的期望步数 \(E_i=\sum_{(i,j)\in G}\frac{E_j+1}{out[i]}\),\(out[i]\)为点\(i\)的出度. 那么对于一个DAG可以直接在 ...
- 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 ...
- BZOJ 3143 游走(贪心+期望+高斯消元)
一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这条边的编号的分 ...
- BZOJ 3143 游走 | 数学期望 高斯消元
啊 我永远喜欢期望题 BZOJ 3143 游走 题意 有一个n个点m条边的无向联通图,每条边按1~m编号,从1号点出发,每次随机选择与当前点相连的一条边,走到这条边的另一个端点,一旦走到n号节点就停下 ...
- BZOJ.3143.[HNOI2013]游走(概率 期望 高斯消元)
题目链接 参考 远航之曲 把走每条边的概率乘上分配的标号就是它的期望,所以我们肯定是把大的编号分配给走的概率最低的边. 我们只要计算出经过所有点的概率,就可以得出经过一条边(\(u->v\))的 ...
随机推荐
- PHP常见算法-面试篇(2)
1.顺序查找 思路分析: 从数组的第一个元素开始一个一个向下查找,如果有和目标一致的元素,查找成功:如果到最后一个元素仍没有目标元素,则查找失败. 代码实现: <?php function se ...
- web2py--------------用web2py写 django的例子 --------建立一个投票应用(1)
按照上一篇我们新建一个名为 polls 的app 然后文件结构如下 然后web2py 会自动向里边添加一些代码. 我们需要剔除一些,如这个 controllers ,defualt.py 的ind ...
- codeforce #339(div2)C Peter and Snow Blower
Peter and Snow Blower 题意:有n(3 <= n <= 100 000)个点的一个多边形,这个多边形绕一个顶点转动,问扫过的面积为多少? 思路:开始就认为是一个凸包的问 ...
- LCD显示方向
一.ILI9341内存到显示地址的映射 本文只讨论“正常显示”,不讨论“垂直滚动显示”模式. 可以看到物理内存被两个指针访问,行指针和列指针,行指针范围从000h到013Fh,列指针范围为0000h到 ...
- TypeScript学习指南第一章--基础数据类型(Basic Types)
基础数据类型(Basic Types) 为了搭建应用程序,我们需要使用一些基础数据类型比如:numbers,strings,structures,boolean等等. 在TypeScript中除了Ja ...
- CoreBluetooth - 中心模式
BLE中心模式流程-coding BLE中心模式流程 - 1.建立中心角色 - 2.扫描外设(Discover Peripheral) - 3.连接外设(Connect Peripheral) - 4 ...
- Linux下eclipse的安装以及配置
在安装好jdk并配置好后,就可以进行eclipse的安装了,其步骤如下: 1.下载eclipse 我所用的eclipse为:eclipse-dsl-juno-SR1-linux-gtk.tar 2. ...
- JS单击隐藏界面元素
1. JS代码 <script type="text/javascript" language="javascript"> // function ...
- USB Type-C接口完美无瑕?小心这五点
今年下半年发布的新手机中,采用USB Type-C接口与传统micro USB接口的手机,所占比例大概是一半对一半.采用Type-C接口的手机大多数都是国产手机,而像三星.摩托罗拉以及索尼等老牌的手机 ...
- Qt刷新机制的一些总结(Qt内部画的时候是相当于画在后台一个对象里,然后在刷新的时候调用bitblt统一画,调用window的api并不会影响到后面的那个对象)
前段时间做过一个界面刷新的优化,遇到的坑比较多,在这里做一点点总结吧. 优化的方案是滚动滚动条的时候用截屏的方式代替界面全部刷新,优化完成后,界面在滚动时效率能提升大概一倍,背景介绍完毕. ...