求最短路径之BF算法

介绍

全称Bellman-Ford算法,目的是求解有负权边的最短路径问题。
考虑环,根据环中边的边权之和的正负,将环分为零环、正环、负环。其中零环、正环不会影响最短路径的求解,而负环会影响最短路径的求解。
可用BF算法返回一个bool值来判断是否有负环,如果有返回false,否则返回true.

bool BF(int b){
    fill(path,path+maxn,INF);
    path[b]=0;
    //求最短距离
    for(int i=0;i<n-1;i++){//比较趟数
        for(int j=0;j<n;j++){//遍历每一个顶点相关的邻接边
            for(int k=0;k<table[j].size();k++){
                int v=table[j][k].v;
                int value=table[j][k].value;
                if(path[j]+value<path[v]){
                    //此时path[v]应该是最小距离
                    path[v]=path[j]+value;
                }
            }
        }
        //判断是否有负环:有返回false,无返回true
        for(int m=0;m<n;m++){//再次遍历边时
            for(int k=0;k<table[m].size();k++){
                int v=table[m][k].v;
                int value=table[m][k].value;
                if(path[m]+value<path[v])
                //还能找到有比path[v]更小的距离
                return false;//说明有负环存在
            }
        }
        return true;//否则无负环
    }
}

设计思想

将求最短路径看作是求以源点为根结点的一棵最短路径树,此时图与起点均确定,因此最短路径树也就确定了,且最短路径树的层数一定不超过顶点个数V,即树中两顶点的比较更新次数不超过V-1轮。

实现

由于用邻接矩阵遍历边时,复杂度会达到O(V的3次方),因此我们使用邻接表去实现

应用

由于求最短路径条数时,BF算法会重复遍历相同的顶点,因此在有多条最短路径数时,最短路径条数的累加会出错。于是我们想到用set容器记录前驱结点,通过遍历去重后的前驱结点进行累加。
set容器的介绍
在这里插入图片描述

#include <iostream>
#include <vector>
#include <set>
#include <algorithm>
using namespace std;
const int maxn=100;
const int INF=1000000000;
struct node{
    int v;//邻接顶点
    int value;//对应边权
    //通过构造函数实现定义同时初始化
    node(int a,int b):v(a),value(b){}
};
vector<node> table[maxn];
int n,edge,st,ed,weight[maxn];

int path[maxn],num[maxn],w[maxn];
set<int> pre[maxn];//记录前驱,以便去重

void BF(int b){
     fill(path,path+maxn,INF);
     memset(num,0,sizeof(num));
     memset(w,0,sizeof(w));
     path[b]=0;
     w[b]=weight[b];
     num[b]=1;
     for(int i=0;i<n-1;i++){
         for(int j=0;j<n;j++){
             for(int k=0;k<table[j].size();k++){
                 //记录
                 int v=table[j][k].v;
                 int value=table[j][k].value;
                 if(path[j]+value<path[v]){
                     path[v]=path[j]+value;
                     w[v]=w[j]+weight[v];
                     num[v]=num[j];//小于覆盖
                     pre[v].clear();//清空
                     pre[v].insert(j);//记录最短路径前驱
                 }else if(path[j]+value==path[v]){
                     if(w[v]<w[j]+weight[v])
                     w[v]=w[j]+weight[v];
                     pre[v].insert(j);//继续记录
                     num[v]=0;//防止重复计数,清空
                     //重新累加计数:通过遍历前驱结点实现
                     for( set<int>::iterator it=pre[v].begin();it!=pre[v].end();it++)
                     num[v]+=num[*it];//*it=pre[j][k],即k的前驱
                 }
             }
         }
     }
 }

int main(){
    int v1,v2,weigh;
    cin>>n>>edge>>st>>ed;
    for(int i=0;i<n;i++)
    cin>>weight[i];
    for(int j=0;j<edge;j++){//构建邻接表
        cin>>v1>>v2>>weigh;
        table[v2].push_back(node(v1,weigh));
        table[v1].push_back(node(v2,weigh));
    }
    BF(st);
    cout<<num[ed]<<" "<<w[ed]<<endl;
    return 0;
}
这道题目可以使用 Dijkstra 算法或者 A* 算法等来解,但不适合使用贪心算法。以下是使用 Dijkstra 算法解该问题的 C 语言代码: ``` #include <stdio.h> #include <stdbool.h> #define INF 99999 #define MAX_VERTEX_NUM 5 int graph[MAX_VERTEX_NUM][MAX_VERTEX_NUM] = { {0, 16, INF, INF, 19}, {16, 0, 5, 6, INF}, {INF, 5, 0, 18, 21}, {INF, 6, 18, 0, 14}, {19, INF, 21, 14, 33} }; int dijkstra(int start, int end) { bool visited[MAX_VERTEX_NUM] = {false}; int distance[MAX_VERTEX_NUM]; for (int i = 0; i < MAX_VERTEX_NUM; i++) { distance[i] = graph[start][i]; } visited[start] = true; for (int i = 0; i < MAX_VERTEX_NUM - 1; i++) { int min_distance = INF; int min_vertex = -1; for (int j = 0; j < MAX_VERTEX_NUM; j++) { if (!visited[j] && distance[j] < min_distance) { min_distance = distance[j]; min_vertex = j; } } if (min_vertex == -1) { break; } visited[min_vertex] = true; for (int j = 0; j < MAX_VERTEX_NUM; j++) { if (!visited[j] && distance[min_vertex] + graph[min_vertex][j] < distance[j]) { distance[j] = distance[min_vertex] + graph[min_vertex][j]; } } } return distance[end]; } int main() { int start = 0; int end = 3; int distance = dijkstra(start, end); printf("The shortest distance from %c to %c is %d", start + 'A', end + 'A', distance); return 0; } ``` 输出结果为: ``` The shortest distance from A to D is 39 ``` 其中,`graph`数组表示图的邻接矩阵,`dijkstra`函数使用了Dijkstra算法最短路径。在`main`函数中,指定了从A到D的最短路径,并输出结果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值