亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C語言求解無向圖頂點之間的所有最短路徑

 更新時間:2019年01月17日 10:34:22   作者:uestcjerry  
這篇文章主要為大家詳細(xì)介紹了C語言求解無向圖頂點之間的所有最短路徑,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實例為大家分享了C語言求解無向圖頂點之間的所有最短路徑的具體代碼,供大家參考,具體內(nèi)容如下

思路一:

DFS,遇到終點之后進(jìn)行記錄
輔助存儲:

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

實現(xiàn):

//查找無向圖的所有最短路徑,直接dfs就可以解決了
//記錄保存這里用 vector<vector<int>> 重新搞一下 OK
// 時間復(fù)雜度 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) {   //遇到終點,進(jìn)行路徑判定
    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);     //存儲當(dāng)前路徑 
      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)

額外開銷:

存儲每一步的路徑,如何維護(hù)好,盡量避免循環(huán)查找。

思路三:

1. 先求出起始點start到其余所有點的最短路徑;  Dijkstra
2. 然后以終點end為開始,反向進(jìn)行dfs/bfs搜索;  
每回退 i 層,判斷值(path-i)與起點到當(dāng)前點最短路徑長度 temp 的比較;
二者相等,則繼續(xù)(利用子問題的正確性); 若 (path-i) < temp ,則這個點不在最短路徑上,放棄。

如圖所示:

先求出start到其余所有點的最短路徑;

然后從 end 點開始往回搜索;

end上面一個點,(path - 1 = 3)等于起始點到它的最短路徑長 3,判斷,是最短路徑上的點,繼續(xù);

再往上搜索:

左邊那個點3,因為此時(path - 2)= 2,而那個點的 temp=3,即 (path - i) < temp ,因此那個點一定不在 start 到 end 的最短路徑上。
而上面那個點2,此時 (path - 2)= 2 , 而那個點 temp = 2, 即 (path - i) == temp , 因此它必然在 start 到 end 的最短路徑上。繼續(xù)搜索下去 。

重復(fù)這樣的過程直到搜索完畢,最終得到兩條最短路徑。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 通過C++程序示例理解設(shè)計模式中的外觀模式

    通過C++程序示例理解設(shè)計模式中的外觀模式

    這篇文章主要介紹了通過設(shè)計模式中的外觀模式及相關(guān)的C++程序示例,外觀模式在高層提供了一個統(tǒng)一的接口實現(xiàn)一定程度上的解耦,需要的朋友可以參考下
    2016-03-03
  • C++ Dijkstra算法之求圖中任意兩頂點的最短路徑

    C++ Dijkstra算法之求圖中任意兩頂點的最短路徑

    這篇文章主要為大家詳細(xì)介紹了用C++經(jīng)典算法-Dijkstra算法求任意兩頂點之間的最短路徑,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • C++使用map實現(xiàn)多進(jìn)程拷貝文件的程序思路

    C++使用map實現(xiàn)多進(jìn)程拷貝文件的程序思路

    這篇文章主要介紹了C++使用mmap實現(xiàn)多進(jìn)程拷貝文件,通過本文給大家分享程序思路及完整代碼,代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • C++解決合并兩個排序的鏈表問題

    C++解決合并兩個排序的鏈表問題

    本文主要介紹了通過C++解決合并兩個排序的鏈表并使新鏈表中的節(jié)點仍然是遞增排序的。文中代碼講解詳細(xì),有需要的朋友可以參考一下
    2021-12-12
  • 基于opencv的行人檢測(支持圖片視頻)

    基于opencv的行人檢測(支持圖片視頻)

    本文主要介紹了基于opencv的行人檢測(支持圖片視頻),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-12-12
  • C語言中使用快速排序算法對元素排序的實例詳解

    C語言中使用快速排序算法對元素排序的實例詳解

    這篇文章主要介紹了C語言中使用快速排序算法對元素排序的實例詳解,文中細(xì)分了幾個情況來舉例,在注釋里有說明,需要的朋友可以參考下
    2016-04-04
  • C++超詳細(xì)講解函數(shù)對象

    C++超詳細(xì)講解函數(shù)對象

    在c++中,我們把所有能當(dāng)作函數(shù)使用的對象統(tǒng)稱為函數(shù)對象。它是實現(xiàn)operator()的任何類型,此運算符被稱為調(diào)用運算符,當(dāng)調(diào)用此操 作符時,其表現(xiàn)形式如同普通函數(shù)調(diào)用一般,因此取名叫函數(shù)對象
    2022-06-06
  • VC下實現(xiàn)fopen支持中文的方法

    VC下實現(xiàn)fopen支持中文的方法

    這篇文章主要介紹了VC下實現(xiàn)fopen支持中文的方法,需要的朋友可以參考下
    2014-07-07
  • C++堆棧類模板實現(xiàn)代碼

    C++堆棧類模板實現(xiàn)代碼

    這篇文章主要為大家詳細(xì)介紹了C++堆棧類模板的實現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-12-12
  • c++ map,mutimap刪除問題分析

    c++ map,mutimap刪除問題分析

    本文詳細(xì)介紹c++ map,mutimap刪除操作時的一些問題,提供了解決方法,需要的朋友可以參考下
    2012-11-11

最新評論