C語言求解無向圖頂點(diǎn)之間的所有最短路徑
本文實(shí)例為大家分享了C語言求解無向圖頂點(diǎn)之間的所有最短路徑的具體代碼,供大家參考,具體內(nèi)容如下
思路一:
DFS,遇到終點(diǎn)之后進(jìn)行記錄
輔助存儲(chǔ):
std::vector<int> tempPath; std::vector<std::vector<int>> totalPath;
實(shí)現(xiàn):
//查找無向圖的所有最短路徑,直接dfs就可以解決了
//記錄保存這里用 vector<vector<int>> 重新搞一下 OK
// 時(shí)間復(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) { //遇到終點(diǎn),進(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); //存儲(chǔ)
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); //存儲(chǔ)當(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;
}時(shí)間分析:
O(V + E)
缺點(diǎn):
可能會(huì)爆棧,我這里算86W點(diǎn)+414W邊直接爆,小的沒問題。
思路二如下:
BFS,位圖/vector/.. 記錄好每一步的路徑即可
時(shí)間
O(V + E)
額外開銷:
存儲(chǔ)每一步的路徑,如何維護(hù)好,盡量避免循環(huán)查找。
思路三:
1. 先求出起始點(diǎn)start到其余所有點(diǎn)的最短路徑; Dijkstra
2. 然后以終點(diǎn)end為開始,反向進(jìn)行dfs/bfs搜索;
每回退 i 層,判斷值(path-i)與起點(diǎn)到當(dāng)前點(diǎn)最短路徑長度 temp 的比較;
二者相等,則繼續(xù)(利用子問題的正確性); 若 (path-i) < temp ,則這個(gè)點(diǎn)不在最短路徑上,放棄。
如圖所示:

先求出start到其余所有點(diǎn)的最短路徑;
然后從 end 點(diǎn)開始往回搜索;
end上面一個(gè)點(diǎn),(path - 1 = 3)等于起始點(diǎn)到它的最短路徑長 3,判斷,是最短路徑上的點(diǎn),繼續(xù);
再往上搜索:
左邊那個(gè)點(diǎn)3,因?yàn)榇藭r(shí)(path - 2)= 2,而那個(gè)點(diǎn)的 temp=3,即 (path - i) < temp ,因此那個(gè)點(diǎn)一定不在 start 到 end 的最短路徑上。
而上面那個(gè)點(diǎn)2,此時(shí) (path - 2)= 2 , 而那個(gè)點(diǎn) temp = 2, 即 (path - i) == temp , 因此它必然在 start 到 end 的最短路徑上。繼續(xù)搜索下去 。
重復(fù)這樣的過程直到搜索完畢,最終得到兩條最短路徑。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
c++ 數(shù)字類型和字符串類型互轉(zhuǎn)詳解
今天小編就為大家分享一篇講解c++ 數(shù)字類型和字符串類型互轉(zhuǎn)的文章,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-09-09
C語言之實(shí)現(xiàn)棧的基礎(chǔ)創(chuàng)建
這篇文章主要介紹了C語言之實(shí)現(xiàn)棧的基礎(chǔ)創(chuàng)建,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C語言控制臺(tái)實(shí)現(xiàn)字符飛機(jī)大戰(zhàn)
這篇文章主要為大家詳細(xì)介紹了C語言控制臺(tái)實(shí)現(xiàn)字符飛機(jī)大戰(zhàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12
詳解利用C語言如何實(shí)現(xiàn)簡單的內(nèi)存池
這篇文章主要給大家介紹了關(guān)于C語言如何實(shí)現(xiàn)簡單的內(nèi)存池的相關(guān)資料,設(shè)計(jì)內(nèi)存池的目標(biāo)是為了保證服務(wù)器長時(shí)間高效的運(yùn)行,通過對(duì)申請(qǐng)空間小而申請(qǐng)頻繁的對(duì)象進(jìn)行有效管理,減少內(nèi)存碎片的產(chǎn)生,合理分配管理用戶內(nèi)存,需要的朋友可以參考下2021-08-08

