畅通工程续 ​​​​​​​(Floyd 和 Dijkstra)

1)Floyd算法   (计算多源点到其他各顶点的最短路径的长度

2)Dijkstra算法(计算单源点到其他各顶点的最短路径的长度

 

 

畅通工程续 

Description

某省自从实行了很多年的畅通工程计划后,终于修建了很多路。不过路多了也不好,每次要从一个城镇到另一个城镇时,都有许多种道路方案可以选择,而某些方案要比另一些方案行走的距离要短很多。这让行人很困扰。
现在,已知起点和终点,请你计算出要从起点到终点,最短需要行走多少距离。

Input

本题目包含多组数据,请处理到文件结束。
每组数据第一行包含两个正整数N和M(0<N<200,0<M<1000),分别代表现有城镇的数目和已修建的道路的数目。城镇分别以0~N-1编号。
接下来是M行道路信息。每一行有三个整数A,B,X(0<=A,B<N,A!=B,0<X<10000),表示城镇A和城镇B之间有一条长度为X的双向道路。
再接下一行有两个整数S,T(0<=S,T<N),分别代表起点和终点。

Output

对于每组数据,请在一行里输出最短需要行走的距离。如果不存在从S到T的路线,就输出-1.

Sample Input

3 3

0 1 1

0 2 3

1 2 1

0 2

3 1

0 1 1

1 2

Sample Output

2

-1

1)Floyd算法 

#include <iostream>
#include <cstdio>
#include <string>
#include <cstring>
#include <cmath>
#include <sstream>
#include <algorithm>
#include <set>
#include <map>
#include <vector>
#include <queue>
#include <iomanip>
#include <stack>

using namespace std;


const int INF = 0x3f3f3f3f;
const int MAXN =205;
int dp[MAXN][MAXN];

int main()
{
    int n, m;//n个点,m条线
   
    int a, b, c;
    while(cin >> n >> m)
    {
        memset(dp, 0x3f, sizeof(dp));//初始化一个较大的值 
        
        int i, j, k;
        for(i = 0;i < n;++i)//一定不要忘这个特殊的初始化 
            dp[i][i] = 0;
            
        for(i = 0;i < m;++i)
        {
            cin >> a >> b >> c;//a到b路长c 
            dp[a][b] = min(dp[a][b], c);//若两点直连多条线,选短的 
            dp[b][a] = dp[a][b];//无向图,故此。 若有向图则不必
        }

        for(k = 0;k < n;++k)
            for(i = 0;i < n;++i)
                for(j = 0;j < n;++j)
                dp[i][j] = min(dp[i][j], dp[i][k] + dp[k][j]);
                
                
        cin >> a >> b;
        if(dp[a][b] != INF)
            cout << dp[a][b] << endl;
        else
            cout << -1 << endl;
    }
}

 2)Dijkstra算法

#include <iostream>
#include <cstdio>
#include <string>
#include <cstring>
#include <cmath>
#include <sstream>
#include <algorithm>
#include <set>
#include <map>
#include <vector>
#include <queue>
#include <iomanip>
#include <stack>
using namespace std;

const int MAXN =205;//节点数目

const int INF = 0x3f3f3f3f;

int dis[MAXN];//源点到其他点的最短距离
int mp[MAXN][MAXN];//所有点之间的距离
int vis[MAXN];//访问节点标记



int main()
{
    int n, m;//n个点,m条线
    int i,j,a, b, c;
    while(cin >> n >> m)
    {
        memset(mp, 0x3f, sizeof(mp));//初始化一个较大的值 
        for(i = 0;i < n;++i)//一定不要忘这个特殊的初始化 
            mp[i][i] = 0;

        

        for(i = 0;i < m;++i)
        {
            cin >> a >> b >> c;//a到b路长c 
            mp[a][b] = min(mp[a][b], c);//若两点直连多条线,选短的 
            mp[b][a] = mp[a][b];//无向图,故此。 若有向图则不必
        }
        
        cin >> a >> b;//输入起点终点a,b 
        memset(dis, 0x3f, sizeof(dis));//初始化一个较大的值 
        for(i = 0;i < n;++i)
            dis[i] = mp[a][i];

        memset(vis, 0, sizeof(vis));//初始化未访问
        vis[a] = 1;

        int v, mi;
        //每次都 vis 一个点,加上上面一次 vis, 所以一共 vis n次
        for(i = 1;i < n;++i)
        {
        //选出当前离 s 最近的点
            mi = INF;
            for(j = 0;j <n;++j)
            {
                if(!vis[j] && dis[j] < mi)
                {
                    mi = dis[j];
                    v = j;
                }
            }
            vis[v] = 1;
        //更新 dis
            for(j = 0;j < n;++j)
                if(!vis[j] && mi + mp[v][j] < dis[j])
                    dis[j] = mi + mp[v][j];
            }           

            if(dis[b]==INF) 
                printf("-1\n");
            else 
                printf("%d\n",dis[b]);
        }
    }

用优先队列优化后的版本:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;

const int INF=0x3f3f3f3f;
const int MAXN=100001;
struct qnode
{
    int v;
    int c;
    qnode(int _v=0,int _c=0):v(_v),c(_c){}
    bool operator <(const qnode &r)const
    {
        return c>r.c;
    }
};
struct Edge
{
    int v,cost;
    Edge(int _v=0,int _cost=0):v(_v),cost(_cost){}
};
vector<Edge> E[MAXN];
bool vis[MAXN];
int dist[MAXN];
void Dijkstra(int n,int start)//点的编号从1开始或者0开始都ok
{
    memset(vis,false,sizeof(vis));
    for(int i=0;i<=n;i++)
        dist[i]=INF;
    priority_queue<qnode> que;
    while(!que.empty())
        que.pop();
    dist[start]=0;
    que.push(qnode(start,0));
    qnode tmp;
    while(!que.empty())
    {
        tmp=que.top();
        que.pop();
        int u=tmp.v;
        if(vis[u])continue;
        vis[u]=true;
        for(int i=0;i<E[u].size();i++)
        {
            int v=E[tmp.v][i].v;
            int cost=E[u][i].cost;
            if(!vis[v]&&dist[v]>dist[u]+cost)
            {
                dist[v]=dist[u]+cost;
                que.push(qnode(v,dist[v]));
            }
        }
    }
}
void addedge(int u,int v,int w)
{
    E[u].push_back(Edge(v,w));
}
int main() {
    // freopen("in.txt","r",stdin); 
    int n, m;
    while(~scanf("%d%d",&n,&m)) {
        int a, b, x;
        for(int i = 0; i <= n; i++) {
            E[i].clear();
        }
        for(int i = 0; i < m; i++) {
            scanf("%d %d %d", &a, &b, &x);
            E[a].push_back(Edge(b,x));
            E[b].push_back(Edge(a,x));
        }
        int s, t;
        scanf("%d %d", &s, &t);
        Dijkstra(n, s);
        if(dist[t] == INF)
            printf("-1\n");
        else 
            printf("%d\n",dist[t]);
    }
    return 0;
}

 

 Dijkstra打印路径的代码详见另一篇博客

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值