【数据结构与算法】最小生成树 | 最短路径(下)

简介: 【数据结构与算法】最小生成树 | 最短路径(下)

👉最短路径👈


路径:在图 G = (V, E) 中,若从顶点 vi 出发有一组边使其可到达顶点 vj,则称顶点 vi 到顶点 vj 的顶点序列为从顶点 vi 到顶点 vj 的路径。

路径长度:对于不带权的图,一条路径的路径长度是指该路径上的边的条数;对于带权的图,一条路径的路径长度是指该路径上各个边权值的总和。


最短路径问题:从在带权有向图 G 中的某一顶点出发,找出一条通往另一顶点的最短路径,最短也就是沿路径各边的权值总和达到最小。单源最短路径问题是给点一个起点,求出起点到其他点的最短路径;而多源最短路径问题就是求出图中任意两点的最多路径。


Dijkstra算法

deb6a73303c1416bb1b0d7a5ce155293.png

99bb9dd0aca64795af6f77976467e613.png11239afdd82849edbb865919bfd24862.png


877d90ef24604318b1b95b2ae34832e2.png


namespace matrix
{
  class Graph
  {
  typedef Graph<V, W, W_MAX, Direction> Self;
  // ...
  public:
    // Dijkstra的时间复杂度为O(N^2),空间复杂度为O(N)
    void Dijkstra(const V& src, vector<W>& dist, vector<int>& pPath)
    {
      size_t srci = GetVertexIndex(src);
      size_t n = _vertexs.size();
      // 初始状态
      dist.resize(n, W_MAX);
      pPath.resize(n, -1);
      dist[srci] = W();
      pPath[srci] = srci;
      // 已经确定最短路径的顶点集合S
      vector<bool> S(n, false);
      for (size_t i = 0; i < n; ++i)
      {
        // 选出未确定最短路径的顶点,用已经确定最短路径的顶点去
        // 更新其他顶点的最短路径
        int u = 0;  // u是已经确定最短路径的顶点(注:存在错位)
        W min = W_MAX;
        for (size_t j = 0; j < n; ++j)
        {
          if (S[j] == false && dist[j] < min)
          {
            u = j;
            min = dist[j];
          }
        }
        S[u] = true;
        // 松弛更新u连接顶点v  srci->u + u->v <  srci->v  更新
        // v是还未确定最短路径的顶点
        for (size_t v = 0; v < n; ++v)
        {
          // Dijkstra算法只能确定没有负权的图的最短路径
          // 原因是没有负权,当前已经确定的最短路径肯定
          // 是该顶点的最短路径。而存在负权的话,可能多
          // 走几个顶点的路径长度会比当前确定的最短路径
          // 的长度还要短。所以Dijkstra算法只能确定没有
          // 负权的图的最短路径
          if (S[v] == false && _matrix[u][v] != W_MAX
            && dist[u] + _matrix[u][v] < dist[v])
          {
            dist[v] = dist[u] + _matrix[u][v];
            pPath[v] = u;
          }
        }
      }
    }
    // 打印最短路径
    void PrintShortPath1(const V& src, const vector<W>& dist, const vector<int>& pPath)
    {
      size_t srci = GetVertexIndex(src);
      size_t n = _vertexs.size();
      for (size_t i = 0; i < n; ++i)
      {
        if (i != srci)
        {
          // 生成从起点srci到顶点i的最短路径
          vector<int> path;
          size_t parenti = i;
          while (parenti != srci)
          {
            path.push_back(parenti);
            parenti = pPath[parenti];
          }
          path.push_back(srci);
          reverse(path.begin(), path.end());
          cout << "从起点" << src << "到顶点" << _vertexs[i] << "的最短路径为" << dist[i] << endl;
          cout << "路径为";
          for (auto index : path)
          {
            cout << _vertexs[index];
            if (index != *(path.end() - 1))
              cout << "->";
          }
          cout << endl << "---------------------------" << endl;
        }
      }
    }
  }
}

d8e538b0b66d4f14ad1bb9b74089e8db.png

Dijkstra 算法无法解决有负权的图


void GraphDijkstraTest2()
{
  // 图中带有负权路径时,贪心策略则失效了。
  // 测试结果可以看到s->t->y之间的最短路径没更新出来
  const char* str = "sytx";
  Graph<char, int, INT_MAX, true> g(str, strlen(str));
  g.AddEdge('s', 't', 10);
  g.AddEdge('s', 'y', 5);
  g.AddEdge('t', 'y', -7);
  g.AddEdge('y', 'x', 3);
  vector<int> dist;
  vector<int> parentPath;
  g.Dijkstra('s', dist, parentPath);
  g.PrintShortPath('s', dist, parentPath);
}

7226e82b5aaf4fa8bf0700fc01d3231a.png

Dijkstra 算法用已经确定最短路径的顶点来更新未确定最短路径的顶点。


BellmanFord算法

1192bf64148b4f208c45dfaf24d42648.png

19a581c7a56a454c8830b6980ddbf7ac.png

25472d4dd1104ce9ad8d7b7ea0a48723.png

BellmanFord 算法的优化是通过队列来优化的,将更新的更短的路径入队列,从而更新包含该路径的路径。优化后,BellmanFord 算法的最好情况是 O(N^2),最坏情况是 O(N^3)。

a3a3a6edc2a9463db96db72363d6a739.png


负权回路问题

ee2079da95d641739103895fb5158211.png


namespace matrix
{
  class Graph
  {
  // ...
  public:
    // BellmanFord算法的时间复杂度为O(N^3),空间复杂度为O(N)
    bool BellmanFord(const V& src, vector<W>& dist, vector<int>& pPath)
    {
      size_t n = _vertexs.size();
      size_t srci = GetVertexIndex(src);
      // vector<W> dist,记录srci-其他顶点最短路径权值数组
      dist.resize(n, W_MAX);
      // vector<int> pPath 记录srci-其他顶点最短路径父顶点数组
      pPath.resize(n, -1);
      // 先更新srci->srci为缺省值
      dist[srci] = W();
      // 总体最多更新n轮
      for (size_t k = 0; k < n; ++k)
      {
        // 顶点i到顶点j 更新一次
        bool update = false;
        //cout << "更新第:" << k << "轮" << endl;
        for (size_t i = 0; i < n; ++i)
        {
          for (size_t j = 0; j < n; ++j)
          {
            // dist[i]为起点srci到i的距离,_matrix[i][j]为i到j的距离
            if (_matrix[i][j] != W_MAX && dist[i] + _matrix[i][j] < dist[j])
            {
              update = true;
              //cout << _vertexs[i] << "->" << _vertexs[j] << ":" << _matrix[i][j] << endl;
              dist[j] = dist[i] + _matrix[i][j];
              pPath[j] = i;
            }
          }
        }
        // 如果这个轮次中没有更新出更短路径,那么后续轮次就不需要再走了
        if (update == false)
        {
          break;
        }
      }
      // 还能更新就是存在负权回路
      for (size_t i = 0; i < n; ++i)
      {
        for (size_t j = 0; j < n; ++j)
        {
          // srci -> i + i ->j
          if (_matrix[i][j] != W_MAX && dist[i] + _matrix[i][j] < dist[j])
          {
            return false;
          }
        }
      }
      return true;  // 不存在负权回路
    }
  }
  void GraphBellmanFordTest()
  {
    const char* str = "syztx";
    Graph<char, int, INT_MAX, true> g(str, strlen(str));
    g.AddEdge('s', 't', 6);
    g.AddEdge('s', 'y', 7);
    g.AddEdge('y', 'z', 9);
    g.AddEdge('y', 'x', -3);
    g.AddEdge('z', 's', 2);
    g.AddEdge('z', 'x', 7);
    g.AddEdge('t', 'x', 5);
    g.AddEdge('t', 'y', 8);
    g.AddEdge('t', 'z', -4);
    g.AddEdge('x', 't', -2);
    // 存在负权回路的样例
    /*const char* str = "syztx";
    Graph<char, int, INT_MAX, true> g(str, strlen(str));
    g.AddEdge('s', 't', 6);
    g.AddEdge('s', 'y', 7);
    g.AddEdge('y', 'z', 9);
    g.AddEdge('y', 'x', -3);
    g.AddEdge('y', 's', 1); // 新增
    g.AddEdge('z', 's', 2);
    g.AddEdge('z', 'x', 7);
    g.AddEdge('t', 'x', 5);
    g.AddEdge('t', 'y', -8); //更改
    g.AddEdge('t', 'z', -4);
    g.AddEdge('x', 't', -2);*/
    vector<int> dist;
    vector<int> parentPath;
    if (g.BellmanFord('s', dist, parentPath))
      g.PrintShortPath('s', dist, parentPath);
    else
      cout << "存在负权回路,无法求出最短路径" << endl;
  }
}

3e86e56f48024583ba9584d9479b44ec.png

6c92ca6a77ab4c58a0d5fff1209c2efd.png


FloydWarshall算法

f49b282a84d244768d29a4444b2bf71f.png

72d1485525a44440a210de788bad67cf.png

8cdc045f9fc14627943702a683a44897.png


即 Floyd 算法本质是三维动态规划,D[i][j][k] 表示从点 i 到点 j 只经过 0 到 k 个点最短路径,然后建立起转移方程,然后通过空间优化,优化掉最后一维度,变成一个最短路径的迭代算法,最后即得到所以点的最短路。

dc147daf82c448dd8bc2f6af56624c4a.png

namespace matrix
{
  class Graph
  {
  // ...
  public:
    // FloydWarshall算法的时间复杂度为O(N^3),空间复杂度为O(N^2)
    void FloydWarshall(vector<vector<W>>& vvDist, vector<vector<int>>& vvpPath)
    {
      size_t n = _vertexs.size();
      vvDist.resize(n);
      vvpPath.resize(n);
      // 初始化权值和路径矩阵
      for (size_t i = 0; i < n; ++i)
      {
        vvDist[i].resize(n, W_MAX);
        vvpPath[i].resize(n, -1);
      }
      // 直接相连的边更新一下
      for (size_t i = 0; i < n; ++i)
      {
        for (size_t j = 0; j < n; ++j)
        {
          if (_matrix[i][j] != W_MAX)
          {
            vvDist[i][j] = _matrix[i][j];
            vvpPath[i][j] = i;
          }
          // 自己到自己的距离为默认值
          if (i == j)
            vvDist[i][j] = W();
        }
      }
      // abcdef:a {中间节点} f 或 b {中间节点} c
      // 最短路径的更新:i->{其他顶点}->j
      for (size_t k = 0; k < n; ++k)
      {
        for (size_t i = 0; i < n; ++i)
        {
          for (size_t j = 0; j < n; ++j)
          {
            // k作为的中间点尝试去更新i->j的路径
            // vvDist[i][j]是从i到j的最短路径的长度
            // vvpPath[i][j]中存的是从i到j路径上与j直接相连的顶点下标
            if (vvDist[i][k] != W_MAX && vvDist[k][j] != W_MAX
              && vvDist[i][k] + vvDist[k][j] < vvDist[i][j])
            {
              vvDist[i][j] = vvDist[i][k] + vvDist[k][j];
              // 找出跟j相连的上一个邻接顶点
              // 如果k和j直接相连.上一个点就是k,vvpPath[k][j]存就是k
              // 如果k和j没有直接相连,k->...->x->j,vvpPath[k][j]存就是x
              // vvpPath[k][j]中存的是从k到j路径上与j直接相连的顶点下标
              vvpPath[i][j] = vvpPath[k][j];
            }
          }
        }
        // 打印权值和路径矩阵观察数据
        /*
        for (size_t i = 0; i < n; ++i)
        {
          for (size_t j = 0; j < n; ++j)
          {
            if (vvDist[i][j] == W_MAX)
            {
              //cout << "*" << " ";
              printf("%3c", '*');
            }
            else
            {
              //cout << vvDist[i][j] << " ";
              printf("%3d", vvDist[i][j]);
            }
          }
          cout << endl;
        }
        cout << endl;
        for (size_t i = 0; i < n; ++i)
        {
          for (size_t j = 0; j < n; ++j)
          {
            //cout << vvParentPath[i][j] << " ";
            printf("%3d", vvpPath[i][j]);
          }
          cout << endl;
        }
        cout << "=================================" << endl;
        */
      }
    }
  }
  void FloydWarShallTest()
  {
    const char* str = "12345";
    Graph<char, int, INT_MAX, true> g(str, strlen(str));
    g.AddEdge('1', '2', 3);
    g.AddEdge('1', '3', 8);
    g.AddEdge('1', '5', -4);
    g.AddEdge('2', '4', 1);
    g.AddEdge('2', '5', 7);
    g.AddEdge('3', '2', 4);
    g.AddEdge('4', '1', 2);
    g.AddEdge('4', '3', -5);
    g.AddEdge('5', '4', 6);
    vector<vector<int>> vvDist;
    vector<vector<int>> vvParentPath;
    g.FloydWarshall(vvDist, vvParentPath);
    // 打印任意两点之间的最短路径
    for (size_t i = 0; i < strlen(str); ++i)
    {
      // 一维数组vvDist[i]是从顶点i到其他点的最短路径的距离
      // 一维数组vvParentPath[i]是从顶点i到其他点的最短路径
      g.PrintShortPath(str[i], vvDist[i], vvParentPath[i]);
      cout << endl;
    }
  }
}


b30087352bbf4f3988afdd56c47ba779.png

f93188f15afd4219abe1015367b131e5.png



总结


Dijkstra 算法只能求出没有负权的图的最短路径,时间复杂度为 O(N^3)。BellmanFord 算法能够求出有负权的图的最短路径,时间复杂度为 O(N^3)。但存在负权回路问题,任何算法都无法解决负权回路问题。Dijkstra 算法和 BellmanFord 算法都需要给点起点,求得的是从起点到其他点的最短路径;而 FloydWarshall 算法能够求出任意两点之间的最短路径,时间复杂度为 O(N^3)。图论中的重点内容是图重要的基本概念、邻接矩阵和邻接表的优缺点、广度优先遍历和深度优先遍历、最小生成树和最短路径等。


👉总结👈


本篇博客主要讲解了最小生成树的 Kruskal 算法和 Prim 算法以及最短路径的 Dijkstra 算法、BellmanFord 算法和 FloydWarshall 算法等。那么以上就是本篇博客的全部内容了,如果大家觉得有收获的话,可以点个三连支持一下!谢谢大家!💖💝❣️

相关文章
|
3月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
107 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
2天前
|
存储 人工智能 算法
【C++数据结构——图】最短路径(头歌教学实验平台习题) 【合集】
任务描述 本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。 相关知识 为了完成本关任务,你需要掌握:Dijkst本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。为了完成本关任务,你需要掌握:Dijkstra算法。带权有向图:该图对应的二维数组如下所示:Dijkstra算法:Dijkstra算法是指给定一个带权有向图G与源点v,求从v到G中其他顶点的最短路径。Dijkstra算法的具体步骤如下:(1)初始时,S只包含源点,即S={v},v的距离为0。
33 15
|
2天前
|
算法 C++
【C++数据结构——图】最小生成树(头歌实践教学平台习题) 【合集】
【数据结构——图】最小生成树(头歌实践教学平台习题)目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:【合集】任务描述 本关任务:编写一个程序求图的最小生成树。相关知识 为了完成本关任务,你需要掌握:1.建立邻接矩阵,2.Prim算法。建立邻接矩阵 上述带权无向图对应的二维数组,根据它建立邻接矩阵,如图1建立下列邻接矩阵。注意:INF表示无穷大,表示整数:32767 intA[MAXV][MAXV];Prim算法 普里姆(Prim)算法是一种构造性算法,从候选边中挑
24 10
|
3月前
|
算法 决策智能
基于prim算法求出网络最小生成树实现网络社团划分和规划
该程序使用MATLAB 2022a版实现路线规划,通过排序节点权值并运用Prim算法生成最小生成树完成网络规划。程序基于TSP问题,采用遗传算法与粒子群优化算法进行路径优化。遗传算法通过编码、选择、交叉及变异操作迭代寻优;粒子群优化算法则通过模拟鸟群觅食行为,更新粒子速度和位置以寻找最优解。
|
5月前
|
算法 定位技术
路径规划算法 - 求解最短路径 - A*(A-Star)算法
路径规划算法 - 求解最短路径 - A*(A-Star)算法
180 1
|
5月前
|
自然语言处理 算法
HanLP — HMM隐马尔可夫模型 - 路径规划算法 - 求解最短路径 - 维特比(Viterbi)算法
HanLP — HMM隐马尔可夫模型 - 路径规划算法 - 求解最短路径 - 维特比(Viterbi)算法
70 0
HanLP — HMM隐马尔可夫模型 - 路径规划算法 - 求解最短路径 - 维特比(Viterbi)算法
|
6月前
|
算法 Java
Java语言实现最短路径算法(Shortest Path)
Java语言实现最短路径算法(Shortest Path)
77 3
|
5月前
|
算法
路径规划算法 - 求解最短路径 - Dijkstra(迪杰斯特拉)算法
路径规划算法 - 求解最短路径 - Dijkstra(迪杰斯特拉)算法
125 0
|
6月前
|
存储 传感器 算法
|
7月前
|
算法 Java
Java数据结构与算法:最短路径算法
Java数据结构与算法:最短路径算法