此图用朴素迪氏或堆优化迪氏都会出错,floyd可以处理。
但floyd无法处理负权环,最短距离是无穷小。在环上不断循环。
经过k条边的最短距离(可能有负权变)
循环k次,循环第i次时,m_vDis表示各点最多经过i-1条边的最短距离;vDis表示各点最多经过i条边的最短距离。
template
class CBellMan
{
public:
CBellMan(int n, const vector
{
m_vDis.assign(n, INF);
m_vDis[s] = 0;
for (int i = 1; i <= k; i++)
{
vector
for (const auto& v : edges)
{
if (INF == m_vDis[v[0]])
{
continue;
}
curDis[v[1]] = min(curDis[v[1]], m_vDis[v[0]] + v[2]);
}
m_vDis.swap(curDis);
}
}
vector
};
#include
#include
using namespace std;
int main()
{
const int INF = 1000 * 1000 * 1000;
vector
vector
for (int i = 0; i < results.size(); i++)
{
CBellMan<> bm(4, edges, 0, i);
for (int j = 0; j < 4; j++)
{
assert(bm.m_vDis[j] == results[i][j]);
}
}
}
最短路径就是经过“点数-1”条边的最短路径。如果没环,这些边可以到达任意点。如果有正权环和0权环,则拿掉这个环。如果负权环,则最小距离是无穷小。下面来检测负权环。循环“点数-1”后,再循环一次,如果有点的最短距离变小,则一定有负权环;没负权环,不会变短。如果有负权环,则再循环一次,一定有点(任意负权环的负权边的终点)距离变短。假定此点是e,拿掉负权环上所有的边后,源点到e的最短路径为Path。不拿掉负权环,则e的最短路径为:Path+此负权环。
template
class CBellMan
{
public:
CBellMan(int n, const vector
{
m_vDis.assign(n, INF);
m_vDis[s] = 0;
for (int i = 1; i <= k; i++)
{
vector
Do(edges, curDis);
m_vDis.swap(curDis);
}
}
bool Check(const vector
{
vector
Do(edges, curDis);
for (int i = 0; i < curDis.size(); i++)
{
if (m_vDis[i] != curDis[i])
{
return true;
}
}
return false;
}
void Do(const std::vector
{
for (const auto& v : edges)
{
if (INF == m_vDis[v[0]])
{
continue;
}
curDis[v[1]] = min(curDis[v[1]], m_vDis[v[0]] + v[2]);
}
}
vector
};
#include
#include
#include "BellMan.h"
using namespace std;
void Test1()
{
const int INF = 1000 * 1000 * 1000;
vector
vector
for (int i = 0; i < results.size(); i++)
{
CBellMan<> bm(4, edges, 0, i);
for (int j = 0; j < 4; j++)
{
assert(bm.m_vDis[j] == results[i][j]);
}
}
}
void Test2()
{
const int INF = 1000 * 1000 * 1000;
vector
vector
for (int i = 0; i < 3; i++)
{
edges[3][2] = -5 - i;
CBellMan<> bm(4, edges, 0, 3);
assert(results[i] == bm.Check(edges));
}
}
int main()
{
Test1();
Test2();
}
源码及测试用例:
https://download.csdn.net/download/he_zhidan/88393784
基础算法的C++实现课程,请点击下面的CSDN学院的链接。
https://edu.csdn.net/course/detail/38771
我还做了其它课程,比如:C++入职培训,C#入职培训。
https://edu.csdn.net/lecturer/6176
Win10 VS2022 Ck++17 或win7 VS2019 C++17
如果你时间宝贵,只想看精华,请到CSDN下载频道下载《闻缺陷则喜算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653
博主有几句话想对大家说 |
算法是程序的灵魂,没有好的算法,程序就死气沉沉。 |
问题发现得越早,越给老板省钱。我简称为:闻缺陷则喜。 |
有所得,以墨记之,故曰墨家 |