单源最短路径——Dijkstra贪心算法

单源最短路径——Dijkstra贪心算法

实现方法:c && c++的STL栈


给定邻接矩阵a和起始节点v,计算出到各个点的最短距离,并用栈式储存给出路径。


代码全文

#include<stdio.h>
#include<iostream>
#include<stack> //数组模拟stack实在是有点麻烦,我唯一使用的c++成分就是栈

using namespace std;

const double MAXNUM = 100000;
#define n 5

void dijkstra(int v, double a[][n+1], double dist[], int prev[]);

void find (int v, int i, int prev[]);

int main() {
    printf("Min Route of 1 start:\n");
    
    int v = 1;

    double a[n + 1][n + 1] = 
    {{0,      0,      0,      0,      0,      0},
     {0, MAXNUM,     10, MAXNUM,     30,    100},
     {0, MAXNUM, MAXNUM,     50, MAXNUM, MAXNUM},
     {0, MAXNUM, MAXNUM, MAXNUM, MAXNUM,     10},
     {0, MAXNUM, MAXNUM,     20, MAXNUM,     60},
     {0, MAXNUM, MAXNUM, MAXNUM, MAXNUM, MAXNUM}};
    
    double dist[6];
    int prev[6];

    dijkstra(v, a, dist, prev);
    for(int i = 1; i <= n; i++) {
        if (i == v) continue;
        printf("From node %d to %d is %lf ", v, i, dist[i]);
        find(v, i, prev);
        printf("\n");
    }

    return 0;
}

void dijkstra(int v, double a[][n+1], double dist[], int prev[]) {
    if(v < 1 || v > n) return; //v是起始节点
    bool s[n + 1];

    //初始化
    for (int i = 1; i <= n; i++) {
        dist[i] = a[v][i];
        s[i] = false;
        if (dist[i] == MAXNUM) prev[i] = 0;
        else prev[i] = v;
    }

    dist[v] = 0;
    s[v] = true;

    for (int i = 1; i < n; i++) {
        double temp = MAXNUM;
        int u = v;
        for (int j = 1; j <= n; j++)
            if ((! s[j]) && (dist[j] < temp)) {
                u = j;
                temp = dist[j];
            }
        s[u] = true;
        for (int j = 1; j <= n; j++)
            if ((! s[j]) && (a[u][j] < MAXNUM)) {
                double newdist = dist[u] + a[u][j];
                if (newdist < dist[j]) {
                    //dist[j] 减少
                    dist[j] = newdist;
                    prev[j] = u;
                }
            }
    }
}

void find(int v, int i, int prev[]) {
    stack<int> s;
    int u = i;
    while (prev[u] != v) {
        s.push(prev[u]);
        u = prev[u];
    }
    printf("%d", v);
    while (! s.empty()) {
        printf("->%d", s.top());
        s.pop();
    }
    printf("->%d", i);
}

运行结果
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值