题目描述:



详细分析:

我们对于编号的分配,很明显可以发现如下的分配就是期望最小的:对经过的期望次数越大的边赋予更小的编号。

那么问题就转化为了怎么求一条边的经过的期望次数,我们发现边数非常大所以肯定不好弄,所以我们就转而看很少的点。因为我们会发现如果我们能知道经过每个点的期望次数,那么这条边的期望次数很轻松就能表达出来。

比如如下的式子:(设 \(ans[i]\) 为经过第 \(i\) 个点的期望次数, \(du[i]\) 为第 \(i\) 个点的度数, \(res[i]\) 为经过第 \(i\) 条边的期望次数)

\[res[i] = \dfrac{ans[from]}{du[from]} + \dfrac{ans[to]}{du[to]}
\]

这个式子应该很好理解,就是说每个点等概率地选择和他相连的边,所以选择这一条边地期望次数就是经过它的期望次数除以它的度数,也就是与他相连的边的数量,因为这条边可以从两个端点开始走然后经过,所以应该加上两个端点的值。

上文探讨了如果知道经过所有点的期望次数如何求经过这条边的期望,那么下文就来看看如何求经过每个点的期望次数。

很明显可以列出这样的一个式子:

\[ans[i] = \sum \dfrac{ans[to]}{du[to]}
\]

注意 \(to\) 是指所有与 \(i\) 有边直接相连的点,\(to\) 不包含 \(n\) 号节点,因为这个式子的含义从是 \(to\) 等概率地回到 \(i\) 节点,可是 \(n\) 号节点就停了,也就不存在再走回来的情况了

但是其实还有一种特殊情况,就是对于 \(1\) 号节点,其作为初始节点所以一定在开始时被经过一次,所以其不仅要计算从别的点到来的期望次数,更要算其一开始的这一次

\[ans[1] = 1 + \sum \dfrac{ans[to]}{du[to]}
\]

我们会发现上文的这个式子会出现循环依赖的情况,就是假设 \(A\) 的值需要 \(B\) 的值才能推出来,但是 \(B\) 也同样需要 \(A\) 才能推出来。而且我们考虑这个式子不含有最大值最小值的操作,所以就考虑使用高斯消元,把这个式子化成一个方程,然后求解.

那么既然要高斯消元就要考虑我们的未知数是什么,我们的系数是什么,常数是什么,这一切都是根据我上面的式子得出来的.

首先未知数肯定非常容易,就是我们不知道的数嘛,那我们不知道什么?就是 \(ans\) 数组啊,所以 \(ans\) 就是我们的未知数, \(ans[i]\) 就代表我们的第 \(i\) 个未知数.

这个明白了之后剩下的就非常简单的,考虑对上面的式子进行转化

\[ans[i] - \sum \dfrac{1}{du[to]} \times ans[to] = 0
\]

很明显 \(du\) 数组我们是知道的,又发现有一个 \(\dfrac{1}{du[to]} \times ans[to]\) 的项,所以 \(du\) 数组就理所应当的成为了我们的系数

会发现了常数项除了 \(ans[1]\) 的方程含有一个 \(1\) ,其他的都是 \(0\).

注意在代码里我的 \(a\) 数组开的二维,因为我们的高斯消元需要知道第几个方程,所以就按照第一个点的顺序给方程编了号,所以 \(a\) 数组的第一维就是编号,第二维才是我们的未知数,这也就与正常的高斯消元一样了

代码详解:

点击查看代码
#include<bits/stdc++.h>
using namespace std;
const double eps = 1e-6;
const int MAXN = 505;
const int MAXM = 130000;
struct edge{
int nxt,to;
edge(int _nxt = 0,int _to = 0){
nxt = _nxt,to = _to;
}
}e[2 * MAXM];
double a[MAXN][MAXN];
double ans[MAXN],res[MAXM];
int from[MAXM],to[MAXM],head[MAXN],du[MAXN];
int n,m,cnt;
void Gauss(int n){ //推荐里面写一个 n ,然后直接按模板敲就好了
for(int i=1; i<=n; i++){
int mx = i;
for(int j=i + 1; j<=n; j++){
if(fabs(a[j][i]) > fabs(a[mx][i])){
mx = j;
}
}
if(mx != i) {
for(int j=1; j<=n+1; j++){
swap(a[i][j],a[mx][j]);
}
}
for(int j=1; j<=n; j++){
if(j != i){
double tmp = a[j][i] / a[i][i];
for(int k=i; k<=n+1; k++){
a[j][k] -= tmp * a[i][k];
}
}
}
}
for(int i=1; i<=n; i++){ //ans[i] 即我们最后解出来的解
ans[i] = a[i][n+1] / a[i][i];
}
}
void add_edge(int from,int to){
e[++cnt] = edge(head[from],to);
head[from] = cnt;
}
int main(){
cin>>n>>m;
for(int i=1; i<=m; i++){
cin>>from[i]>>to[i];
add_edge(from[i],to[i]);
add_edge(to[i],from[i]);
du[from[i]]++;du[to[i]]++;
}
for(int i=1; i<n; i++){ //注意一点,因为到 n 就停了,所以 n 的期望等就不用算了
a[i][i] = 1; //根据我们的式子可以化出来
for(int j=head[i]; j; j = e[j].nxt){
int to = e[j].to;
if(to != n){ //注意 to != n
a[i][to] = -1.0/du[to]; //根据我们的式子可以化出来
}
}
}
a[1][n] = 1; //根据式子可以化出来
Gauss(n-1);
for(int i=1; i<=m; i++){
if(from[i] != n){
res[i] += ans[from[i]] / du[from[i]]; //到达这个点的期望除以度数,就是从这个点到当前点的期望
}
if(to[i] != n){
res[i] += ans[to[i]] / du[to[i]];
}
}
sort(res+1,res+m+1); //从小到大排边
//一定要相信 STL,如果自己写一个 cmp(例如我),就会造成神奇的精度问题
double h = 0;
for(int i=1; i<=m; i++){
//越大的期望的边给他越小的编号
h += res[i] * (m - i + 1);
}
printf("%.3f",h);
return 0;
}

我的上文对这个代码的解释应该也比较合理了,如果有任何疑问或者感觉我说的不对的地方欢迎大家留言或者私信我

【题解】金牌导航-高斯消元/Luogu P3232 游走的更多相关文章

  1. HDU4870_Rating_双号从零单排_高斯消元求期望

    原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=4870 原题: Rating Time Limit: 10000/5000 MS (Java/Other ...

  2. 【bzoj3601】一个人的数论 莫比乌斯反演+高斯消元

    题目描述 题解 莫比乌斯反演+高斯消元 (前方高能:所有题目中给出的幂次d,公式里为了防止混淆,均使用了k代替) #include <cstdio> #include <cstrin ...

  3. 【bzoj2115】[Wc2011] Xor DFS树+高斯消元求线性基

    题目描述 输入 第一行包含两个整数N和 M, 表示该无向图中点的数目与边的数目. 接下来M 行描述 M 条边,每行三个整数Si,Ti ,Di,表示 Si 与Ti之间存在 一条权值为 Di的无向边. 图 ...

  4. 【bzoj4184】shallot 线段树+高斯消元动态维护线性基

    题目描述 小苗去市场上买了一捆小葱苗,她突然一时兴起,于是她在每颗小葱苗上写上一个数字,然后把小葱叫过来玩游戏. 每个时刻她会给小葱一颗小葱苗或者是从小葱手里拿走一颗小葱苗,并且 让小葱从自己手中的小 ...

  5. luogu P2962 [USACO09NOV]灯Lights 高斯消元

    目录 题目链接 题解 题目链接 luogu P2962 [USACO09NOV]灯Lights 题解 可以折半搜索 map合并 复杂度 2^(n / 2)*logn 高斯消元后得到每个点的翻转状态 爆 ...

  6. 2017年中国大学生程序设计竞赛-中南地区赛暨第八届湘潭市大学生计算机程序设计大赛题解&源码(A.高斯消元,D,模拟,E,前缀和,F,LCS,H,Prim算法,I,胡搞,J,树状数组)

    A------------------------------------------------------------------------------------ 题目链接:http://20 ...

  7. BZOJ 1013 | 一份写了一堆注释的高斯消元题解

    题意 给出\(n\)维直角坐标系中\(n + 1\)个点的坐标,它们都在一个\(n\)维球面上,求球心坐标. 题解 设球面上某两个点坐标为\((a_1, a_2, ... a_n)\)和\((b_1, ...

  8. 【Luogu】P3317重建(高斯消元+矩阵树定理)

    题目链接 因为这个专门跑去学了矩阵树定理和高斯消元qwq 不过不是很懂.所以这里只放题解 玫葵之蝶的题解 某未知dalao的矩阵树定理 代码 #include<cstdio> #inclu ...

  9. Luogu P3389 高斯消元

    https://www.luogu.com.cn/problem/P3389 主元消元法[模板] 高斯消元是解决多元线性方程组的方法,再学习它之前,先引入一个东西--行列式 行列式的性质: 这里我们只 ...

随机推荐

  1. Spring Boot 需要独立的容器运行吗?

    可以不需要,内置了 Tomcat/ Jetty 等容器.

  2. 利用事件的冒泡特性,为子标签添加Onclick事件

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  3. ElasticSearch-学习笔记04-复杂查询

    service package com.huarui.service; import com.huarui.entity.SearchEntity; import com.huarui.exectio ...

  4. CSS3渐变效果工具

    推荐一个css3渐变效果工具,觉得有帮助的可以收藏下. 工具链接 CSS3 渐变(gradients)可以让你在两个或多个指定的颜色之间显示平稳的过渡.CSS3 定义了两种类型的渐变(gradient ...

  5. About HTML

    HTML 简介 HTML 历史 最初的 HTMl 是由 CERN负责制定的,后来转交给 IETF. 在 1990-1995 年期间, HTML 经历了许多次的版本修改与扩充: 1995 年的时候 HT ...

  6. 记离线缓存(manifest)一大坑,断定其只适用于静态网站或离线应用

    今天看了离线缓存(manifest)方面的资料,兴冲冲地就想给自己的网站用上.待我把代码都写好部署上服务器,并测试过OK的时候,在SegmentFault刷了一把manifest方面的问答,才发现这个 ...

  7. background-clip 和 background-origin

    下面都是我自己的理解, 如果有不对的地方, 还请大家帮忙指出.下面是在 chrome 浏览器上测试的 background-clip 和 background-origin 先说说background ...

  8. tomcat 安装配置及问题解决

    最近没写程序 刚想运行一个jsp程序发现tomcat出现一些问题,然后就重新装了程序,重新配置 总结经验就是不要怕报错,把错误复制下来,百度里面都有解决办法 要安装与自己jdk版本相匹配的tomcat ...

  9. 嵌入式框架iframe

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  10. spring-基于注解的aop开发(快速入门)

    步骤: 1.导入坐标 <dependency> <groupId>junit</groupId> <artifactId>junit</artif ...