【概率dp 高斯消元】bzoj3270: 博物馆
一类成环概率dp的操作模式
Description
Input
Output
Sample Input
1 2
0.5
0.5
Sample Output
HINT
对于100%的数据有 n <= 20,n-1 <= m <= n(n-1)/2
题目分析
记$f[i][j]$为两人分别在$i$和$j$的概率,$u,v$为分别与$i,j$相连的点,那么有$$f[i][j]= \begin{cases} \sum \frac{f[u][v]*(1-p[u])*(1-p[v]))}{deg[u]*deg[v]}& \text{(u!=i&&v!=j)}\\\sum \frac{f[u][v]*(1-p[u])*p[v])}{deg[u]*deg[v]}& \text{(u!=i&&v=j)}\\\sum \frac{f[u][v]*p[u]*(1-p[v]))}{deg[u]*deg[v]}& \text{(u=i&&v!=j)}\\\sum \frac{f[u][v]*(1-p[u]*(1-p[v]))}{deg[u]*deg[v]}& \text{(u=i&&v=j)}\end{cases}$$
由于这里dp的关系成环,不能够直接转移,所以要使用高斯消元。由此共有$n^2$个状态,即$n^2$个未知量。故用$id[i][j]$来表示$f[i][j]$,就可方便地把这些未知量的关系用矩阵表示出来。
需要注意的是,初始$f[S][T]$的概率为1,不过后面又会有经过它的概率,所以在计算时需要考虑其初始概率。这里算是一个概率dp的奇怪的地方:最终$f[S][T]>1$。
再者是处理的一个小技巧:添一条自边便于处理。
#include<bits/stdc++.h>
const int maxn = ; int n,m,tot,S,T,deg[maxn],id[maxn][maxn];
double out[maxn],p[maxn],ans[maxn],mp[maxn][maxn];
int G[maxn][maxn]; void Gauss(int n)
{
double bse;
for (int i=, r; i<=n; i++)
{
r = i;
for (int j=i+; j<=n; j++)
if (fabs(mp[j][i]) > fabs(mp[r][i])) r = j;
if (r!=i) std::swap(mp[i], mp[r]);
bse = mp[i][i];
for (int j=i; j<=n+; j++) mp[i][j] /= bse;
for (int j=i+; j<=n; j++)
{
bse = mp[j][i];
for (int k=i; k<=n+; k++)
mp[j][k] -= bse*mp[i][k];
}
}
ans[n] = mp[n][n+];
for (int i=n-; i; i--)
{
ans[i] = mp[i][n+];
for (int j=i+; j<=n; j++) ans[i] -= ans[j]*mp[i][j];
}
}
int main()
{
scanf("%d%d%d%d",&n,&m,&S,&T);
tot = n*n;
for (int i=,x,y; i<=m; i++)
{
scanf("%d%d",&x,&y), ++deg[x], ++deg[y];
G[x][++G[x][]] = y, G[y][++G[y][]] = x;
}
for (int i=; i<=n; i++)
{
scanf("%lf",&p[i]);
G[i][++G[i][]] = i;
out[i] = (1.0-p[i])/(1.0*deg[i]);
}
for (int i=, t=; i<=n; i++)
for (int j=; j<=n; j++)
id[i][j] = ++t;
mp[id[S][T]][tot+] = -;
for (int i=; i<=n; i++)
for (int j=; j<=n; j++)
{
--mp[id[i][j]][id[i][j]];
for (int l=; l<=G[i][]; l++)
for (int r=; r<=G[j][]; r++)
{
int u = G[i][l], v = G[j][r];
if (u==v) continue;
if (u==i&&v==j) mp[id[i][j]][id[i][j]] += p[i]*p[j];
if (u!=i&&v!=j) mp[id[i][j]][id[u][v]] += out[u]*out[v];
if (u==i&&v!=j) mp[id[i][j]][id[u][v]] += p[u]*out[v];
if (u!=i&&v==j) mp[id[i][j]][id[u][v]] += out[u]*p[v];
}
}
Gauss(tot);
for (int i=; i<=n; i++) printf("%.6lf ",ans[id[i][i]]);
return ;
}
看到有网上有些博客说,高斯消元这里$mp[id[i][j]][id[u][v]]$是指$id[i][j]$状态转移到$id[u][v]$状态的概率,这个说法其实是不对的。$id[i][j]$实际上不过是和常规高斯消元一样,代表处理第几条方程;至于$id[u][v]$则表示当前这条方程中的$f[u][v]$;所以$mp[id[i][j]][id[u][v]]$是指第$id[i][j]$条方程中,第$id[u][v]$个未知量的系数。
END
【概率dp 高斯消元】bzoj3270: 博物馆的更多相关文章
- BZOJ3270 博物館 概率DP 高斯消元
BZOJ3270 博物館 概率DP 高斯消元 @(XSY)[概率DP, 高斯消元] Description 有一天Petya和他的朋友Vasya在进行他们众多旅行中的一次旅行,他们决定去参观一座城堡博 ...
- BZOJ_1778_[Usaco2010 Hol]Dotp 驱逐猪猡_概率DP+高斯消元
BZOJ_1778_[Usaco2010 Hol]Dotp 驱逐猪猡_概率DP+高斯消元 题意: 奶牛们建立了一个随机化的臭气炸弹来驱逐猪猡.猪猡的文明包含1到N (2 <= N <= 3 ...
- LightOJ - 1151概率dp+高斯消元
概率dp+高斯消元 https://vjudge.net/problem/LightOJ-1151 题意:刚开始在1,要走到100,每次走的距离1-6,超过100重来,有一些点可能有传送点,可以传送到 ...
- 【bzoj1778】[Usaco2010 Hol]Dotp 驱逐猪猡 矩阵乘法+概率dp+高斯消元
题目描述 奶牛们建立了一个随机化的臭气炸弹来驱逐猪猡.猪猡的文明包含1到N (2 <= N <= 300)一共N个猪城.这些城市由M (1 <= M <= 44,850)条由两 ...
- BZOJ 3270: 博物馆 [概率DP 高斯消元]
http://www.lydsy.com/JudgeOnline/problem.php?id=3270 题意:一张无向图,一开始两人分别在$x$和$y$,每一分钟在点$i$不走的概率为$p[i]$, ...
- 【BZOJ3640】JC的小苹果 概率DP+高斯消元
[BZOJ3640]JC的小苹果 Description 让我们继续JC和DZY的故事. “你是我的小丫小苹果,怎么爱你都不嫌多!” “点亮我生命的火,火火火火火!” 话说JC历经艰辛来到了城市B,但 ...
- BZOJ 3270 博物馆 ——概率DP 高斯消元
用$F(i,j)$表示A在i,B在j的概率. 然后很容易列出转移方程. 然后可以高斯消元了! 被一个问题困扰了很久,为什么起始点的概率要加上1. (因为其他博客上都是直接写成-1,雾) 考虑初始状态是 ...
- 【概率DP/高斯消元】BZOJ 2337:[HNOI2011]XOR和路径
2337: [HNOI2011]XOR和路径 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 682 Solved: 384[Submit][Stat ...
- BZOJ 3640: JC的小苹果 [概率DP 高斯消元 矩阵求逆]
3640: JC的小苹果 题意:求1到n点权和\(\le k\)的概率 sengxian orz的题解好详细啊 容易想到\(f[i][j]\)表示走到i点权为j的概率 按点权分层,可以DP 但是对于\ ...
随机推荐
- c# Array或List有个很实用的ForEach方法,可以直接传入一个方法对集合中元素操作
using System; using System.Collections.Generic; namespace demo { class Program { static void Main(st ...
- ZROI WC Round1 题解
ZROI WC Round1 题解 Problem A 题意 一个 \(n \times m\) 格子图,一个人从左上角出发,每次向右或者向下走一格,方法如下: 如果他在最下面一排,那么他会往右行走. ...
- java中实现定时任务 task 或quartz
转载大神的 https://www.cnblogs.com/hafiz/p/6159106.html https://www.cnblogs.com/luchangyou/p/6856725.html ...
- NET?.NET Framework?.NET Core?
什么是.NET?什么是.NET Framework?什么是.NET Core? https://www.cnblogs.com/1996V/p/9037603.html 什么是.NET?什么是.NET ...
- JS中实现JSON对象和JSON字符串之间的相互转换
对于主流的浏览器(比如:firefox,chrome,opera,safari,ie8+),浏览器自己提供了JSON对象,其中的parse和stringify方法实现了JSON对象和JSON字符串之间 ...
- Java @Validated 遇到的大坑
我在一个Controller内,在两个方法内使用@Validated,这是两个POST方法会进入的方法,这两个方法的实体类的命名(下图红框内容)不能一样,一样的话就会导致第二个在页面显示不出来错误信息 ...
- AJPFX深入理解之abstract class和interface的区别
含有abstract修饰符的class即为抽象类,abstract 类不能创建的实例对象.含有abstract方法的类必须定义为abstract class,abstract class类中的方法不必 ...
- springboot集成shiro实现身份认证
github地址:https://github.com/peterowang/shiro pom文件 <dependencies> <dependency> <group ...
- swift基础-3
fallthrough 贯穿 case 可以不必写break 如果不需要知道区间内 每一项的值 可以使用 下划线 —— 来代替变量名 忽略 对该值的访问 for index in 1...5{ p ...
- 1068 乌龟棋 2010年NOIP全国联赛提高组
1068 乌龟棋 2010年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 题目描述 Descrip ...