C语言求解无向图顶点之间的所有最短路径

 更新时间:2019年01月17日 10:34:22   作者:uestcjerry  
这篇文章主要为大家详细介绍了C语言求解无向图顶点之间的所有最短路径,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了C语言求解无向图顶点之间的所有最短路径的具体代码,供大家参考,具体内容如下

思路一:

DFS,遇到终点之后进行记录
辅助存储:

std::vector<int> tempPath;
std::vector<std::vector<int>> totalPath;

实现:

//查找无向图的所有最短路径,直接dfs就可以解决了
//记录保存这里用 vector<vector<int>> 重新搞一下 OK
// 时间复杂度 O(N + E)
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <vector>
#include <set>

#define MAX 10
#define INF 999999

int graph[MAX + 1][MAX + 1];
int N, M;          //node, edge
int nodeBook[MAX + 1];
int minPath = INF;
std::vector<int> pathNodeVec;
std::vector<std::vector<int>> allShortVec;
int startNode, endNode;

void dfs(int i, int step)
{
  if (i == endNode) {   //遇到终点,进行路径判定
    if (step < minPath) {
      std::cout << "step < minpath.., size = " << allShortVec.size() << std::endl;
      minPath = step;
      pathNodeVec.push_back(i);
      for (auto &elem : pathNodeVec)
        std::cout << elem << "\t";
      std::cout << std::endl;

      std::vector<int> tempVec = pathNodeVec;
      allShortVec.clear();            //清空
      allShortVec.push_back(tempVec);      //存储
      pathNodeVec.pop_back();
    } else if (step == minPath) {
      std::cout << "step == minpath.., size = " << allShortVec.size() << std::endl;
      pathNodeVec.push_back(i);
      for (auto &elem : pathNodeVec)
        std::cout << elem << "\t";
      std::cout << std::endl;
      std::vector<int> tempVec = pathNodeVec;
      allShortVec.push_back(tempVec);     //存储当前路径 
      pathNodeVec.pop_back();
    } else { ;}
    return;
  }

  nodeBook[i] = 1;
  pathNodeVec.push_back(i);
  for (int x = 1; x <= N; x++) {   //尝试所有可能性
    if (x == i)
      continue;
    if (nodeBook[x] == 1)
      continue;
    if (graph[i][x] == INF)
      continue;
    dfs(x, step + 1);
  }
  nodeBook[i] = 0;
  pathNodeVec.pop_back();
  return ;
}
int main(void)
{
  std::cin >> N >> M;
  for (int x = 1; x <= N; x++)
    nodeBook[x] = 0;    //表示还没有访问
  for (int x = 1; x <= N; ++x)
    for (int y = 1; y <= N; ++y) {
      if (x == y)
        graph[x][y] = 0;
      else
        graph[x][y] = INF;
    }
  for (int i = 1; i <= M; ++i) {
    int tempX, tempY, tempWeight;
    std::cin >> tempX >> tempY >> tempWeight;
    graph[tempX][tempY] = tempWeight;
  }
  std::cout << "please input start node & end node :" << std::endl;
  std::cin >> startNode >> endNode;
  pathNodeVec.clear();
  allShortVec.clear();

  dfs(startNode, 0);
  std::cout << "all shortest path: \t";
  std::cout << "size = " << allShortVec.size() << std::endl;

  for (std::vector<std::vector<int>>::const_iterator it = allShortVec.begin(); it != allShortVec.end(); it++) {
    for (std::vector<int>::const_iterator it2 = (*it).begin(); it2 != (*it).end(); it2++)
      std::cout << (*it2) << "\t";
    std::cout << std::endl;
  }
  getchar();
  return 0;
}

时间分析:

O(V + E)

缺点:

可能会爆栈,我这里算86W点+414W边直接爆,小的没问题。

思路二如下:

BFS,位图/vector/.. 记录好每一步的路径即可

时间

O(V + E)

额外开销:

存储每一步的路径,如何维护好,尽量避免循环查找。

思路三:

1. 先求出起始点start到其余所有点的最短路径;  Dijkstra
2. 然后以终点end为开始,反向进行dfs/bfs搜索;  
每回退 i 层,判断值(path-i)与起点到当前点最短路径长度 temp 的比较;
二者相等,则继续(利用子问题的正确性); 若 (path-i) < temp ,则这个点不在最短路径上,放弃。

如图所示:

先求出start到其余所有点的最短路径;

然后从 end 点开始往回搜索;

end上面一个点,(path - 1 = 3)等于起始点到它的最短路径长 3,判断,是最短路径上的点,继续;

再往上搜索:

左边那个点3,因为此时(path - 2)= 2,而那个点的 temp=3,即 (path - i) < temp ,因此那个点一定不在 start 到 end 的最短路径上。
而上面那个点2,此时 (path - 2)= 2 , 而那个点 temp = 2, 即 (path - i) == temp , 因此它必然在 start 到 end 的最短路径上。继续搜索下去 。

重复这样的过程直到搜索完毕,最终得到两条最短路径。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • c++ 数字类型和字符串类型互转详解

    c++ 数字类型和字符串类型互转详解

    今天小编就为大家分享一篇讲解c++ 数字类型和字符串类型互转的文章,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-09-09
  • C语言之实现栈的基础创建

    C语言之实现栈的基础创建

    这篇文章主要介绍了C语言之实现栈的基础创建,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++扑克牌的洗牌发牌游戏设计

    C++扑克牌的洗牌发牌游戏设计

    这篇文章主要为大家详细介绍了C++扑克牌的洗牌发牌游戏设计,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • C语言控制台实现字符飞机大战

    C语言控制台实现字符飞机大战

    这篇文章主要为大家详细介绍了C语言控制台实现字符飞机大战,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-12-12
  • 详解利用C语言如何实现简单的内存池

    详解利用C语言如何实现简单的内存池

    这篇文章主要给大家介绍了关于C语言如何实现简单的内存池的相关资料,设计内存池的目标是为了保证服务器长时间高效的运行,通过对申请空间小而申请频繁的对象进行有效管理,减少内存碎片的产生,合理分配管理用户内存,需要的朋友可以参考下
    2021-08-08
  • C/C++内存管理基础与面试

    C/C++内存管理基础与面试

    本章主要介绍C语言与C++的内存管理,以C++的内存分布作为引入,介绍C++不同于C语言的内存管理方式(new delete对比 malloc free),感兴趣的朋友来看看吧
    2022-07-07
  • C语言用fstat函数获取文件的大小方法

    C语言用fstat函数获取文件的大小方法

    今天小编就为大家分享一篇关于C语言用fstat函数获取文件的大小方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • C语言实现简单的计算器

    C语言实现简单的计算器

    这篇文章主要为大家详细介绍了C语言实现简单的计算器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-02-02
  • QT编写简单登录界面的实现示例

    QT编写简单登录界面的实现示例

    登陆界面是网页中常见的界面,本文主要介绍了QT编写简单登录界面的实现示例,具有一定的参考价值,感兴趣的可以了解一下
    2024-02-02
  • C语言版三子棋游戏

    C语言版三子棋游戏

    这篇文章主要为大家详细介绍了C语言版三子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08

最新评论