CodeForces 295B Greg and Graph

题解

\(Floyd\) 算法是一种基于动态规划的算法,以此题为例介绍最短路算法中的 \(Floyd\) 算法。

我们考虑给定一个图,要找出 \(i\) 号点到 \(j\) 号点的最短路径。则该最短路径只有两种可能:

  • \(i\) 号点直接到达 \(j\) 号点的路径中产生最短路径

  • \(i\) 号点经过一些中间点到达 \(j\) 号点的路径中产生最短路径

我们添加一个点 \(k\),使得 \(i\) 号点到 \(j\) 号点再添加后产生的最短路径比添加前的最短路径更短,则称该操作为松弛

下面我们以从一个示例开始介绍算法的步骤。

示例

给定一个赋权有向图,如下图:

我们声明一个集合 \(S\),并称该集合为中转集合

我们声明一个 \(dist\) 数组,其中 \(dist[i][j]\) 表示从源点 \(i\) 号点只经过中转集合中的点,到达目标点 \(j\) 号点的最短路径。

每一轮更新都更新 \(dist\),直到集合 \(S\) 为图的顶点集 \(V\)。

初始化更新

\(S\) 初始化为:

\[S = \{ \}
\]

\(dist\) 数组根据我们的定义,此时 \(dist[i][j]\) 表示不经过图中的任何顶点,直接从源点 \(i\) 号点到达 \(j\) 号点的最短距离,为了方便书写,我们此处采用图的邻接矩阵表示。

\[dist[i][j] = graph[i][j] \quad \forall i,j \in V
\]

故 \(dist\) 被初始化为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & inf & inf \\
inf & 0 & inf & 5 & 15 \\
inf & 30 & 0 & 20 & inf \\
10 & inf & inf & 0 & 25 \\
40 & inf & inf & inf & 0
\end{bmatrix}
\]

第一轮更新

\(S\) 更新为:

\[S = \{ 0 \}
\]

\(dist\) 数组根据我们的定义,此时 \(dist[i][j]\) 表示由图中 \(0\) 号点中转,直接从源点 \(i\) 号点到达 \(j\) 号点的最短距离。该最短路径有两种可能:

  1. 最短路径经过 \(0\) 号点;
  2. 最短路径不经过 \(0\) 号点;

对于第 \(1\) 种情况说明 \(0\) 号点可以松弛之前的最短距离。这两种情况可以统一表示为下式:

\[\begin{align}
dist[i][j] = \min \left( dist[i][j],dist[i][0] + dist[0][j] \right) \quad \forall i,j \in V \tag{1}
\end{align}
\]

此处我们应该注意一下更新的顺序问题,上式表达的含义应该是指只由上一轮的 \(dp[i][j]\)、\(dp[i][0]\) 和 \(dp[0][j]\) 来更新出这一轮的 \(dp[i][j]\)。更准确地说应该可以扩展下数组 \(dist\) 的维度。 如下图所示:

其中 \(k\) 表示这是第 \(k\) 轮更新。按我们的意思,应该写成:

\[dist[i][j][1] = \min \left( dist[i][j][0],dist[i][0][0] + dist[0][j][0] \right) \quad \forall i,j \in V
\]

即只由第 \(0\) 轮的 \(dp[i][j]\)、\(dp[i][0]\) 和 \(dp[0][j]\) 来更新出第 \(1\) 轮的 \(dp[i][j]\)。那我们为什么可以写成式 \(1\) 的样子?原因在于 \(dp[i][0]\) 和 \(dp[0][j]\) 在此轮更新循环中,始终都没有被更新。后面会更详细地说明这个问题。

故 \(dist\) 被更新为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & inf & inf \\
inf & 0 & inf & 5 & 15 \\
inf & 30 & 0 & 20 & inf \\
10 & 15* & 20* & 0 & 25 \\
40 & 45* & 50* & inf & 0
\end{bmatrix}
\]

第二轮更新

\(S\) 更新为:

\[S = \{ 0, 1 \}
\]

\(dist\) 数组根据我们的定义,此时 \(dist[i][j]\) 表示从源点 \(i\) 号点出发,只经过中转集合 \(S\) 中的点,到达 \(j\) 号点的最短距离。该最短路径有两种可能:

  1. 最短路径经过 \(1\) 号点;
  2. 最短路径不经过 \(1\) 号点;

对于第 \(1\) 种情况说明 \(1\) 号点可以松弛之前的最短距离。这两种情况可以统一表示为下式:

\[dist[i][j] = \min \left( dist[i][j],dist[i][1] + dist[1][j] \right) \quad \forall i,j \in V
\]

我们接着就要问了,为什么经过 \(1\) 号点的最短路径距离等于 \(dist[i][1] + dist[1][j]\)。在第一轮更新的时候容易想到,但这轮更新(第二轮更新)中,该问题需要仔细思考一下。

经过 \(1\) 号点的路径这样表示:\(i\)(源点) \(\to\) \(1\) \(\to\) \(j\)(目标点)。而经过 \(1\) 号点的最短路径一定是由 \(i\) 到 \(1\) 的最短路径和 \(1\) 到 \(j\) 的最短路径构成(反证法即可得)。

我们应用这个性质,重新表述为:经过 \(1\) 号点的从源点 \(i\) 出发,由 \(S - \{1\}\) 集合中转到达目标点 \(j\) 的最短路径一定是由从源点 \(i\) 出发,由 \(S - \{1\}\) 集合中转到达目标点 \(1\) 的最短路径从源点 \(1\) 出发,由 \(S - \{1\}\) 集合中转到达目标点 \(j\) 的最短路径构成(正确性显然,只是由原来的范围全路径集合缩小为\(S - \{1\}\) 集合中转的那些路径构成的集合)。

而此轮更新仍然不会改变 \(dist[i][1]\) 和 \(dist[1][j]\) 的值,因此不用关心更新顺序的问题。

综上,上式完全正确。故 \(dist\) 应该被更新为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & 10* & 20* \\
inf & 0 & inf & 5 & 15 \\
inf & 30 & 0 & 20 & 45* \\
10 & 15 & 20 & 0 & 25 \\
40 & 45 & 50 & 50* & 0
\end{bmatrix}
\]

第三轮更新

\(S\) 更新为:

\[S = \{ 0, 1, 2\}
\]

状态转移方程为:

\[dist[i][j] = \min \left( dist[i][j],dist[i][2] + dist[2][j] \right)
\]

状态转移方程的正确性证明完全同上,而由归纳法容易知道对之后的更新该状态转移方程仍然正确。而该轮仍然不会改变 \(dist[i][2]\) 和 \(dist[2][j]\) 的值,因此不用关心更新顺序的问题。

故 \(dist\) 应该被更新为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & 10 & 20 \\
inf & 0 & inf & 5 & 15 \\
inf & 30 & 0 & 20 & 45 \\
10 & 15 & 20 & 0 & 25 \\
40 & 45 & 50 & 50 & 0
\end{bmatrix}
\]

第四轮更新

\(S\) 被更新为:

\[S = \{ 0, 1, 2, 3 \}
\]

\(dist\) 被更新为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & 10 & 20 \\
15* & 0 & 25* & 5 & 15 \\
30* & 30 & 0 & 20 & 45 \\
10 & 15 & 20 & 0 & 25 \\
40 & 45 & 50 & 50 & 0
\end{bmatrix}
\]

第五轮更新

\(S\) 被更新为:

\[S = \{ 0, 1, 2, 3, 4 \}
\]

\(dist\) 被更新为:

\[dist =
\begin{bmatrix}
0 & 5 & 10 & 10 & 20 \\
15 & 0 & 25 & 5 & 15 \\
30 & 30 & 0 & 20 & 45 \\
10 & 15 & 20 & 0 & 25 \\
40 & 45 & 50 & 50 & 0
\end{bmatrix}
\]

此时 \(S\) 集合等于图的顶点集 \(V\)。停止迭代。

最后得到的 \(dist[i][j]\) 就表示源点 \(i\) 到达目标点 \(j\) 的最短路径距离。

程序设计

为了能清晰严谨的说明,我们可以将 \(dist[i][j]\) 设成 \(dist[i][j][k]\)。每次循环用 \(dist[i][j][k - 1]\) 来更新 \(dist[x][y][k]\)(即用上一层的 \(dist\) 来更新这一层的 \(dist\))。但实际上,我们发现可以通过减少一个维度来减小空间复杂度,虽然这时我们要注意更新顺序的问题,但可以进一步发现每次迭代 \(dist[i][m]\) 和 \(dist[m][j]\) 都没有发生被更新为新值,即对应的 \(dist\) 的第 \(m\) 行和第 \(m\) 列都没有发生变化,这可以使得更新顺序可以任意。

如下图展示了在某一次迭代中更新 \(dist[3][4]\) 的过程,对应的公式描述为:

\[dist[3][4] = \min (dist[3][4], dist[3][1] + dist[1][4])
\]

上图表示那次迭代过程中,第 \(1\) 行和第 \(1\) 列的值都没有被更新为新值,正是由此,明显可以看出 \(dist[i][j]\) 的更新顺序确实可以任意。

程序的主要部分可以写出来:

  1. // n : 表示图的顶点数
  2. for (int k = 0; k < n; ++k) {
  3. for (int i = 0; i < n; ++i) {
  4. for (int j = 0; j < n; ++j) {
  5. dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]);
  6. }
  7. }
  8. }

最短路算法对比

算法 \(Floyd\) \(Dijkstra\) \(Bellman\)-\(Ford\)
空间复杂度 \(O\)\((V^2)\) \(O\)\((E)\) \(O\)\((E)\)
时间复杂度 \(O\)\((V^3)\) 看具体实现 \(O\)\((VE)\)
负权边时是否可以处理 可以 不能 可以
判断是否存在负权回路 不能 不能 可以

其中 \(V\) 表示图的顶点数,\(E\) 表示图的边数。

运用 \(Floyd\) 算法解决该题

题目大意:每次删去一个点,求当前删去该点之前,所有点对最短路径之和。

在每删除一个点前,题目明显要求求出的最短路径不是单源,而是多源的。具体地说,第一,针对所有当前存在的点对;第二,对每一对点对求最短路径。

每次删去一个点,直到把所有点都删尽。倒过来就相当于每次加入一个点,而恰好 \(Floyd\) 算法的建立过程中 \(dist\) 数组便记录了这些信息,所以逆序使用 \(Floyd\) 算法即可。

程序:

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. #include<iostream>
  5. #include<vector>
  6. #include<queue>
  7. using namespace std;
  8. typedef pair<int, int> PII;
  9. #define LL long long
  10. int n, m;
  11. int graph[505][505]; // 邻接矩阵存图
  12. int del[505]; // 删除点的序号
  13. LL sum[505]; // 删除点对前,点对的最短距离和
  14. int dist[505][505];
  15. int main()
  16. {
  17. // 输入图的邻接矩阵
  18. cin >> n;
  19. for (int i = 1; i <= n; ++i) {
  20. for (int j = 1; j <= n; ++j) {
  21. cin >> m;
  22. graph[i][j] = m;
  23. }
  24. }
  25. // 输入图的顶点删除顺序
  26. for (int i = 1; i <= n; ++i) cin >> del[i];
  27. // 初始化 dist 数组
  28. memset(dist, 0x3f, sizeof(dist));
  29. for (int i = 1; i <= n; ++i) {
  30. for (int j = 1; j <= n; ++j) {
  31. dist[i][j] = graph[i][j];
  32. }
  33. }
  34. // 根据删除的点的序号,我们逆向添加点,运行Floyd算法
  35. int add[505];
  36. memset(add, 0, sizeof(add));
  37. for (int k = n; k >= 1; --k) {
  38. add[del[k]] = 1;
  39. for (int i = 1; i <= n; ++i) {
  40. for (int j = 1; j <= n; ++j) {
  41. dist[i][j] = min(dist[i][j], dist[i][del[k]] + dist[del[k]][j]);
  42. }
  43. }
  44. for (int i = 1; i <= n; ++i) {
  45. for (int j = 1; j <= n; ++j) {
  46. if (add[i] && add[j]) sum[k] += dist[i][j];
  47. }
  48. }
  49. }
  50. for (int k = 1; k <= n; ++k) printf("%I64d ", sum[k]);
  51. printf("\n");
  52. return 0;
  53. }

ACM - 最短路 - CodeForces 295B Greg and Graph的更多相关文章

  1. 那些年我们写过的三重循环----CodeForces 295B Greg and Graph 重温Floyd算法

    Greg and Graph time limit per test 3 seconds memory limit per test 256 megabytes input standard inpu ...

  2. CodeForces 295B Greg and Graph (floyd+离线)

    <题目链接> 题目大意:给定$n$个点的有向完全带权图$(n\leq500)$,现在进行$n$次操作,每次操作从图中删除一个点(每删除一个点,都会将与它相关联的边都删除),问你每次删点之前 ...

  3. Codeforce 295B Greg and Graph(Floyd的深入理解)

    题目链接:http://codeforces.com/problemset/problem/295/B 题目大意:给出n个点的完全有权有向图,每次删去一个点,求删掉该点之前整张图各个点的最短路之和(包 ...

  4. [CodeForces - 296D]Greg and Graph(floyd)

    Description 题意:给定一个有向图,一共有N个点,给邻接矩阵.依次去掉N个节点,每一次去掉一个节点的同时,将其直接与当前节点相连的边和当前节点连出的边都需要去除,输出N个数,表示去掉当前节点 ...

  5. 295B - Greg and Graph (floyd逆序处理)

    题意:给出任意两点之间的距离,然后逐个删除这些点和与点相连的边,问,在每次删除前的所有点对的最短距离之和 分析:首先想到的是floyd,但是如果从前往后处理,复杂度是(500)^4,超时,我们从后往前 ...

  6. Codeforces 295 B. Greg and Graph

    http://codeforces.com/problemset/problem/295/B 题意: 给定一个有边权的有向图.再给定一个1~n的排列. 按排列中的顺序依次删除点,问每次删除后,所有点对 ...

  7. Codeforces 459E Pashmak and Graph(dp+贪婪)

    题目链接:Codeforces 459E Pashmak and Graph 题目大意:给定一张有向图,每条边有它的权值,要求选定一条路线,保证所经过的边权值严格递增,输出最长路径. 解题思路:将边依 ...

  8. ural 1091. Tmutarakan Exams 和 codeforces 295 B. Greg and Graph

    ural 1091 题目链接:http://acm.timus.ru/problem.aspx?space=1&num=1091 题意是从1到n的集合里选出k个数,使得这些数满足gcd大于1 ...

  9. ACM学习历程—CodeForces 601A The Two Routes(最短路)

    题目链接:http://codeforces.com/problemset/problem/601/A 题目大意是有铁路和陆路两种路,而且两种方式走的交通工具不能在中途相遇. 此外,有铁路的地方肯定没 ...

随机推荐

  1. 【C# 异常处理】调试器 管理异常

    装载自:https://docs.microsoft.com/zh-cn/visualstudio/debugger/managing-exceptions-with-the-debugger?vie ...

  2. RabbitMQ安装以及简单操作应用(针对Windows和C#)

    1.RabbitMQ安装 1.1下载并安装Erlang https://www.erlang.org/downloads 一直点next就安装好了.我直接使用了默认的安装目录.否则的话,应该需要配置一 ...

  3. WCF学习笔记——Day1:一个WCF demo

    Visual Studio2017,使用IIS托管.文中涉及一些WCF的基本概念,e.g.服务契约.托管等.可以先阅读<WCF服务编程>第一章. 1.新建一个WCF服务库(WCF Serv ...

  4. 接口java.util.Map的四个实现类HashMap Hashtable LinkedHashMap TreeMap

    java中HashMap,LinkedHashMap,TreeMap,HashTable的区别 :java为数据结构中的映射定义了一个接口java.util.Map;它有四个实现类,分别是HashMa ...

  5. QT:中文字符串与“常量中有字符串”报错

    解决方法参照: (10条消息) Qt5.9 win7系统 中文字符串报错:常量中有字符串_Be busy living or busy dying-CSDN博客 主要是用QStringLiteral( ...

  6. Flask 自建扩展

    自建扩展介绍 Flask扩展分两类 纯功能, 如: Flask-Login 提供用户认证 对已有的库和工具包装(简化继承操作,并提供有用的功能,更方便) 如: Flask-SQLAlchemy 包装了 ...

  7. Tableau学习Step6一如何制作炫彩地图

    Tableau学习Step6一如何制作炫彩地图 本文首发于博客冰山一树Sankey,去博客浏览效果更好. 一.统计地图概述 1.1 统计地图的基本概念 统计地图的本质:数据的正确对应 将数据信息和地理 ...

  8. Drools 规则引擎应用

    规则引擎-drools 1 .场景 1.1需求 商城系统消费赠送积分 100元以下, 不加分 100元-500元 加100分 500元-1000元 加500分 1000元 以上 加1000分 .... ...

  9. 字符集编码(四):UTF

    在前面文章<字符集编码(中):Unicode>中我们聊了 Unicode 标准并提到其有三种实现形式:UTF-16.UTF-8 和 UTF-32,本篇我们就具体聊聊这三种 UTF 是怎么实 ...

  10. LabVIEW,控件快捷菜单,温度转换

    目前正在自学LabVIEW,深感困难重重,我将偶尔发表一些自己的收获,自认为算是干货了, 搜到这篇随笔的朋友们或多或少遇到了些许困难,希望这能帮助到你们. 内容:练习使用LabVIEW中的控件快捷菜单 ...