热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

bzoj3143游走

Description:一个无向连通图,顶点从1编号到N,边从1编号到M。小Z在该图上进行随机游走,初始时小Z在1号顶点&#

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

要用高斯消元做。
定义f[i]为期望的到点i的次数,du[i]为点i的度数。那么有
f[i]=∑j与i相邻,j≠nf[j]du[j]+[i==1]f[i] = \sum_{j与i相邻, j\neq n} \frac{f[j]}{du[j]} + [i==1]f[i]=ji,j=ndu[j]f[j]+[i==1]
注意点n只进不出,所以它对其他点的到达次数没有贡献。
点1一开始就有一次到达。

我们现在有了n个n元一次方程。由于这个方程不规则,所以我们只能用数值解法来解这个方程。这里采用高斯消元法。

把f解出来后,就把走每条边的期望次数g[e]求出来。
g[e]=f[e.u]du[e.u]+f[e.v]du[e.v]g[e] = \frac{f[e.u]}{du[e.u]} + \frac{f[e.v]}{du[e.v]}g[e]=du[e.u]f[e.u]+du[e.v]f[e.v]

然后从大到小排个序,从1到m标号即可。

代码:

#include
#include
#include
#include
#include
#include using namespace std;#define MAXN 511
#define MAXM 250011typedef pair<int, int> pii;const double eps &#61; 1e-8;struct GRAPH {vector<vector<int> > s;void ClearEdges() {for (auto& i : s) {i.resize(0);}}void Init(int n) {ClearEdges();s.resize(n &#43; 1);}void AddUndi(int u, int v) {s[u].push_back(v);s[v].push_back(u);}
};struct MATRIX {vector<vector<double> > s;MATRIX() {}MATRIX(size_t a, size_t b) {resize(a, b);}inline size_t row() const {return s.size();}inline size_t col() const {return s.at(0).size();}void resize(size_t a, size_t b) {s.resize(a);for (size_t i &#61; 0; i < a; &#43;&#43;i)s[i].resize(b);}void clear() {for (auto& i : s)for (auto& j : i)j &#61; 0;}void swap_row(size_t i, size_t j, size_t k &#61; 0) {for (; k < col(); &#43;&#43;k)swap(s[i][k], s[j][k]);}//s[i] -&#61; s[j] * dvoid sub_row(size_t i, size_t j, double d, size_t k &#61; 0) {for (; k < col(); &#43;&#43;k)s[i][k] -&#61; d * s[j][k];}//O(n^3)void ToUpper(MATRIX& b) {for (size_t i &#61; 0; i < row(); &#43;&#43;i) {double maxv &#61; fabs(s[i][i]);size_t mr &#61; i;for (size_t j &#61; i &#43; 1; j < row(); &#43;&#43;j) {if (maxv < fabs(s[j][i])) {maxv &#61; fabs(s[j][i]);mr &#61; j;}}swap_row(i, mr, i);b.swap_row(i, mr);if (maxv < eps) continue;for (size_t j &#61; i &#43; 1; j < row(); &#43;&#43;j) {double d &#61; s[j][i] / s[i][i];sub_row(j, i, d, i);b.sub_row(j, i, d);}}}
};//ax &#61; b
MATRIX solve_destory(MATRIX& a, MATRIX& b) {a.ToUpper(b);for (int i &#61; a.row() - 1; i; --i) {assert(fabs(a.s[i][i]) > eps);b.s[i][0] /&#61; a.s[i][i];for (int j &#61; 0; j < i; &#43;&#43;j) {b.s[j][0] -&#61; b.s[i][0] * a.s[j][i];}}return b;
}int main() {GRAPH g;static pii es[MAXM];static double gs[MAXN];int n, m;scanf("%d%d", &n, &m);g.Init(n);for (int i &#61; 0; i < m; &#43;&#43;i) {scanf("%d%d", &es[i].first, &es[i].second);--es[i].first;--es[i].second;}for (int i &#61; 0; i < m; &#43;&#43;i) {g.AddUndi(es[i].first, es[i].second);}MATRIX b(n, 1);b.clear();b.s[0][0] &#61; 1;MATRIX a(n, n);for (int i &#61; 0; i < n; &#43;&#43;i) {a.s[i][i] &#61; 1;for (int v : g.s[i]) {if (v !&#61; n-1)a.s[i][v] &#61; -1.0 / g.s[v].size();}}b &#61; solve_destory(a, b);b.s[n-1][0] &#61; 0;for (int i &#61; 0; i < m; &#43;&#43;i) {gs[i] &#61; b.s[es[i].first][0] / g.s[es[i].first].size() &#43; b.s[es[i].second][0] / g.s[es[i].second].size();}sort(gs, gs &#43; m, greater<double>());double ans &#61; 0;for (int i &#61; 0; i < m; &#43;&#43;i) {ans &#43;&#61; gs[i] * (i &#43; 1);}printf("%.3f\n", ans);return 0;
}


推荐阅读
  • vue使用
    关键词: ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • 本文介绍了UVALive6575题目Odd and Even Zeroes的解法,使用了数位dp和找规律的方法。阶乘的定义和性质被介绍,并给出了一些例子。其中,部分阶乘的尾零个数为奇数,部分为偶数。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • C++中的三角函数计算及其应用
    本文介绍了C++中的三角函数的计算方法和应用,包括计算余弦、正弦、正切值以及反三角函数求对应的弧度制角度的示例代码。代码中使用了C++的数学库和命名空间,通过赋值和输出语句实现了三角函数的计算和结果显示。通过学习本文,读者可以了解到C++中三角函数的基本用法和应用场景。 ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了最长上升子序列问题的一个变种解法,通过记录拐点的位置,将问题拆分为左右两个LIS问题。详细讲解了算法的实现过程,并给出了相应的代码。 ... [详细]
author-avatar
粪想升或_519
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有