题意

一个无向连通图,顶点从1编号到N,边从1编号到M。 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这条边的编号的分数。当小Z 到达N号顶点时游走结束,总分为所有获得的分数之和。 现在,请你对这M条边进行编号,使得小Z获得的总分的期望值最小。(2<=N<=500)

分析

直接算边的期望会很大,考虑先算点的期望。

设 $E(i)$ 为经过第 $i$ 个点的期望次数,$D(i)$ 为 $i$ 的度数,设 $v$ 为与 $u$ 相连的点,则

$$E(u) = \sum_{v, v \neq n} \frac{E(v)}{D(v)}$$

从而,经过一条边 $(u, v)$ 的期望次数为

$$E(u, v)= \frac{E(u)}{D(u)} + \frac{E(v)}{D(v)}$$

有两点需要注意,结点1期望步数需要加1(可假设有结点0,以概率1转移到结点1);由于到结点 $n$ 就结束了,不要考虑来自 $n$ 的步数。

由于是无向连通图,不用考虑0和无穷大,方程组有唯一的解。

既然求出每条边的期望次数,当然是给次数大的分配小编号,次数小的分配大编号。

#include<bits/stdc++.h>
using namespace std; const int maxn = +;
typedef double Matrix[maxn][maxn]; //要求系数矩阵可逆
//这里的A是增广矩阵,即A[i][n] 是第i个方程右边的常数bi
//运行结束后A[i][n] 是第i个未知数的值
void gauss_elimination(Matrix A, int n)
{
int i, j, k, r; for(i = ;i < n;i++) //消元过程
{
//选绝对值一行r并与第i行交换
r = i;
for(j = i+; j < n;j++)
if(fabs(A[j][i] > fabs(A[r][i]))) r = j;
if(r != i) for(j = ;j <= n;j++) swap(A[r][j], A[i][j]); //与第i+1~n行进行消元
for(k = i+; k < n;k++)
{
double f = A[k][i] / A[i][i];
for(int j = i;j <= n;j++) A[k][j] -= f * A[i][j]; //已经是阶梯型矩阵了,所以从i开始
}
} //回代过程
for(i = n-;i >= ;i--)
{
for(j = i+; j < n;j++)
A[i][n] -= A[j][n] * A[i][j];
A[i][n] /= A[i][i];
}
} void debug_print(Matrix A, int n)
{
for(int i = ;i < n;i++)
for(int j = ;j <= n;j++) printf("%f%c", A[i][j], j == n ? '\n' : ' ');
} int n, m;
vector<int>edges[maxn];
int d[maxn];
Matrix A;
vector<double>e; int main()
{
scanf("%d%d", &n, &m);
for(int i = ;i < m;i++)
{
int u, v;
scanf("%d%d", &u, &v);
u--; v--; //改成从0开始编号
edges[u].push_back(v);
edges[v].push_back(u);
d[u]++; d[v]++;
} //构造方程组
for(int i = ;i <n;i++)
{
A[i][i] = ;
for(int j = ;j <edges[i].size();j++)
if(edges[i][j] != n-) A[i][edges[i][j]] -= 1.0/d[edges[i][j]];
if(i == ) A[i][n] = ;
} //debug_print(A, n); gauss_elimination(A, n); for(int i = ;i < n;i++)
{
for(int j = ;j < edges[i].size();j++)
{
double tmp = ;
if(i != n-) tmp += A[i][n]/d[i];
if(edges[i][j] != n-) tmp += + A[edges[i][j]][n]/d[edges[i][j]]; //不是终点时
e.push_back(tmp);
}
} sort(e.begin(), e.end()); // for(int i = 0;i < e.size();i++) printf("%f ", e[i]);
// printf("\n"); double res = ;
for(int i = ;i < m;i++)
{
res += e[*i]*(m-i); //无向边重复了一次,所以隔一个取一个
}
printf("%.3f\n", res);
}

参考链接:https://www.luogu.org/problemnew/solution/P3232

P3232 [HNOI2013]游走——无向连通图&&高斯消元的更多相关文章

  1. BZOJ_3143_[Hnoi2013]游走_期望DP+高斯消元

    BZOJ_3143_[Hnoi2013]游走_期望DP+高斯消元 题意: 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机 ...

  2. BZOJ 3143: [Hnoi2013]游走 概率与期望+高斯消元

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

  3. luoguP3232 [HNOI2013]游走 贪心 + 概率期望 + 高斯消元

    首先,题目中的无向简单连通图代表着没有自环,重边... 总分的期望 = 每条边的期望之和...................每条边的期望又可以拆成$u \to v$的期望和$v \to u$的期望 ...

  4. bzoj 3143 [Hnoi2013]游走(贪心,高斯消元,期望方程)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3143 [题意] 给定一个无向图,从1走到n,走过一条边得到的分数为边的标号,问一个边的 ...

  5. P3232 [HNOI2013]游走 解题报告

    P3232 [HNOI2013]游走 题目描述 一个无向连通图,顶点从\(1\)编号到\(N\),边从\(1\)编号到\(M\). 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概 ...

  6. 洛谷P3232 [HNOI2013]游走(高斯消元+期望)

    传送门 所以说我讨厌数学……期望不会高斯消元也不会……好不容易抄好了高斯消元板子被精度卡成琪露诺了…… 首先,我们先算出走每一条边的期望次数,那么为了最小化期望,就让大的期望次数乘上小编号 边的期望次 ...

  7. 题解 P3232 [HNOI2013]游走

    洛谷P3232[NOI2013]游走 题目描述 给定一个 n 个点 m 条边的无向连通图,顶点从 1 编号到 n,边从 1 编号到 m. 小 Z 在该图上进行随机游走,初始时小 Z 在 1 号顶点,每 ...

  8. 洛谷P3232[HNOI2013]游走

    有一个无向简单连通图,顶点从 \(1\) 编号到 \(n\),边从 \(1\) 编号到 \(m\) 小Z在该图上进行随机游走,初始时小Z在\(1\)号顶点,每一步小Z以相等的概率随机选 择当前顶点的某 ...

  9. [bzoj3143] [洛谷P3232] [HNOI2013] 游走

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

随机推荐

  1. redis源码分析(五)--cluster(集群)结构

    Redis集群 Redis支持集群模式,集群中可以存在多个master,每个master又可以拥有多个slave.数据根据关键字映射到不同的slot,每一个master负责一部分的slots,数据被存 ...

  2. scala学习遇到的坑

    1:软件是idea,在同一个包中,类名一样,会导致在调用类的时候,程序无法找到正确的类,最后出错. 2:return坑,在方法中,返回值的类型已经指定了,所以可以放心使用return,但是在函数中,没 ...

  3. jQuery基础事件处理

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. Docker 安装入门 Centos Linux安装Docker 部署mysql

    这次购买了阿里云云服务器,并且安装了Centos 7.5 学习使用Docker, 确认版本信息 Docker 运行在 CentOS 7 上,要求系统为64位.系统内核版本为 3.10 以上. Dock ...

  5. windows操作系统更改 <远程桌面> 端口号

    windows远程桌面连接默认使用的是3389端口,为了避免被他人扫描从而暴力破解远程服务器或者病毒入侵.可以将默认端口修改为其它端口,如8888,11111等.最好修改为10000以后的端口,这样可 ...

  6. .NET core Quartz 定时任务框架 demo

    开始先建个空的web项目. 创建一个新类 QuartzFactory 狠狠的复制就完事了. public class QuartzFactory : IJobFactory { private rea ...

  7. 手把手教你打造高效的 Kubernetes 命令行终端

    Kubernetes 作为云原生时代的操作系统,熟悉和使用它是每名用户的必备技能.本文将介绍一些提高操作 Kubernetes 效率的技巧以及如何打造一个高效的 Kubernetes 命令行终端的方法 ...

  8. Python的字符串函数

    今天用了将近一天的时间去学习Python字符串函数 上午学了17个,下午学了23个(共计40) 详细内容请见菜鸟教程--Python3字符串--Python的字符串内建函数

  9. SpringBoot的入门程序

    1. 创建一个springboot工程 可以参考springboot入门程序 2. 创建一个实体类 @Data //想相当于@Setter.@Getter和@ToString替代了setter.get ...

  10. python3基础之“术语表(2)”

    51.编程: 让计算机执行的指令. 52.代码: 让计算机执行的命令. 53.底层编程语言: 与高级语言相比,更接近二进制的语言. 54.高级编程语言: 读起来像英语的易于理解的语言. 55.汇编语言 ...