最短路径 -- spfa

最短路径 – spfa

Google的第二轮,第一题就是一道图论题目,对于图论薄弱的我,简直是当头一棒!
看题戳这里

此题略复杂,不过看过大牛的解题报告之后,深感自己弱爆了。
大牛的解题报告戳这里

此题还是单源最短路问题,只是分成了24小时的动态路径,而由于题目中的限制条件:

It is guaranteed that
Cost[t] ≤ Cost[t+1]+1 (0 ≤ t ≤ 22) and Cost[23] ≤ Cost[0]+1.

使得我们熟知的最短路算法可以得到正确的解(详细解释参看大牛的报告)

借此机会,学习了一下 spfa 算法
参考资料戳这里

感谢以上引用的各位博主!


附上通过代码(C++)

#include <stdio.h>
#include <limits.h>
#include <string.h>
#include <queue>

#define IMAX (INT_MAX/2)
#define min(a,b) (((a) < (b)) ? (a) : (b))

using namespace std;

int T, Ti, N, M, K, D, S, x, y, t, i, j;
int cost[501][501][24], dis[501][24], vis[501];
//cost 记录输入路径
//dis  以0~23小时开始的单源最短路的距离
//vis  spfa算法的中间变量,记录访问状态


//初始化数组
void init() {
    for (i = 0; i <= N; i++) {
        for (j = 0; j <= N; j++) {
            for (t = 0; t < 24; t++) {
                cost[i][j][t] = IMAX;
            }
        }
    }

    for (i = 0; i <= N; i++) {
        for (t = 0; t < 24; t++) {
            cost[i][i][t] = 0;
        }
    }

    memset(dis, 0, sizeof(dis));
    for (i = 2; i <= N; i++) {
        for (t = 0; t < 24; t++) {
            dis[i][t] = IMAX;
        }
    }
}

int main() {
    scanf("%d", &T);
    for (Ti = 1; Ti <= T; Ti++) {
        scanf("%d %d %d", &N, &M, &K);
        init();
        //处理输入
        for (i = 0; i < M; i++) {
            scanf("%d %d", &x, &y);
            for (j = 0; j < 24; j++) {
                scanf("%d", &t);
                cost[x][y][j] = min(cost[x][y][j], t);
                cost[y][x][j] = min(cost[y][x][j], t);
            }
        }

        //对24小时的每个小时做一遍
        for (t = 0; t < 24; t++) {
            // spfa算法
            queue<int> que;
            que.push(1);
            memset(vis, 0, sizeof(vis));
            vis[1] = 1;
            while (!que.empty()) {
                int pre = que.front();
                for (i = 1; i <= N; i++) {
                    if (cost[pre][i][0] == IMAX) continue;

                    int tm = (t + dis[pre][t]) % 24;
                    //计算时间差,适应动态路径的权重

                    if (dis[i][t] > dis[pre][t] + cost[pre][i][tm]) {
                        dis[i][t] = dis[pre][t] + cost[pre][i][tm];
                        if (vis[i] == 0) {
                            vis[i] = 1;
                            que.push(i);
                        }
                    }
                }
                que.pop();
                vis[pre] = 0;
            }
        }

        //O(1)时间查找输出
        printf("Case #%d:", Ti);
        for (i = 0; i < K; i++) {
            scanf("%d %d", &D, &S);
            if (dis[D][S] == IMAX) {
                printf(" -1");
            }
            else {
                printf(" %d", dis[D][S]);
            }
        }
        printf("\n");
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值