日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

单源最短路SPFA

發布時間:2023/12/19 编程问答 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 单源最短路SPFA 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

單源最短路SPFA

算法思想:

將遍歷到且滿足條件的點加入隊列中,如果遍歷到的點在隊列,就不加入。

對每個結點進行遍歷時,更改其是否在隊列的狀態。

在隊列內即為true

若不在隊列內且其值被更改,則加入隊列

  • if(minx[s,t]>minx[s,k]+minx[k,t])
  • minx[s][t]=minx[s,k]+minx[k,t] if(st==false){st=trueq.push(t) }

    每次更新一個點的最小值后,才加入隊列,借更新的值更新其他點。

    若不更新,則不必加入隊列。

    此外有多種優化方式,最簡單的就是SLF優化,將小權值的點加入隊列最前方,對于數據量小的情況時間優化不大。

    因此時3在隊列中,因此其無法從負環計算

    當4出隊列,3進行遍歷,則可進行對負數查找,隨后將4與2加入隊列,又遍歷2

    若存在負環,則負環值永遠無法累加

    而因更新值時才會加入隊列,因此不存在跳不出循環的情況

    例題:

    代碼如下:

    #include<iostream> #include<algorithm> #include<string.h> #include<stdio.h> #include<string> #include<queue> #include<stack> #include<vector>using namespace std;int n, m; int a, b, c; vector<int> sic[1005]; int bq[1005][1005]; bool st[1005]; int ans[1005];void SPFA() {queue<int>q;q.emplace(1);memset(ans, 0x3f, sizeof ans);st[1] = 1;ans[1] = 0;while (!q.empty()) {int k = q.front();q.pop();st[k] = 0;//將移除隊列的結點拿出for (auto x : sic[k]) {if (ans[x] > ans[k] + bq[k][x]) {ans[x] = ans[k] + bq[k][x];//更新權值if (!st[x]) {q.push(x);//將不再隊列中的滿足條件的加入隊列st[x] = 1;}}}} }int main() {while (cin >> m >> n, n != 0 && m != 0) {memset(bq, 0, sizeof bq);memset(st, 0, sizeof st);memset(sic, 0, sizeof sic);for (int i = 0; i < n; i++) {cin >> a >> b >> c;sic[a].push_back(b);sic[b].push_back(a);bq[a][b] = bq[b][a] = c;}SPFA();cout << ans[m] << endl;}return 0; }

    此題目加入SLF優化后時間變化不大,代碼如下:

    #include<iostream> #include<algorithm> #include<string.h> #include<stdio.h> #include<string> #include<queue> #include<stack> #include<vector>using namespace std;int n, m; int a, b, c; vector<int> sic[1005]; int bq[1005][1005]; bool st[1005]; int ans[1005];void SPFA() {deque<int>q;q.emplace_back(1);memset(ans, 0x3f, sizeof ans);st[1] = 1;ans[1] = 0;while (!q.empty()) {int k = q.front();q.pop_front();st[k] = 0;for (auto x : sic[k]) {if (ans[x] > ans[k] + bq[k][x]) {ans[x] = ans[k] + bq[k][x];if (!st[x]) {if (!q.empty()) {//不為空,則進行判斷對比if (ans[x] > ans[q.front()])q.emplace_front(x);elseq.emplace_back(x);}else {q.emplace_back(x);}st[x] = 1;}}}} }int main() {while (cin >> m >> n, n != 0 && m != 0) {memset(bq, 0, sizeof bq);memset(st, 0, sizeof st);memset(sic, 0, sizeof sic);for (int i = 0; i < n; i++) {cin >> a >> b >> c;sic[a].push_back(b);sic[b].push_back(a);bq[a][b] = bq[b][a] = c;}SPFA();cout << ans[m] << endl;}return 0; }

    總結

    以上是生活随笔為你收集整理的单源最短路SPFA的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。