传送门:点击打开链接
题意:给你有向图,每条边呈周期性开放,即开放a时间,再关闭b时间,再开放a时间以此类推
如果时间不足以穿过这条路则不能走,你可以在节点等待时间,问从s走到t所需要的最小时间
仔细想一想这题,对于某条边,越早走到u点一定是最优的,大不了我就等时间嘛
所以,这只是一个普通的dijistra然后稍微在每一条边的距离上做了点手脚而已
在节点等待时间,我们可以认为是某一条路的长度增加了,所以只要在读取边长度的时候,稍微处理一下等待时间,就可以了
需要注意的地方:
1.如果开放时间小于边的长度,这条边是无论怎样都不能通过的,所以一开始就不能把这条边添加进去
2.如果到达某一点时,这条边是开放的,但是剩下的开放时间已经不足以通过这条边,那么要等到关闭后,重新开放时,才能通过
#include<map>
#include<set>
#include<cmath>
#include<stack>
#include<queue>
#include<cstdio>
#include<string>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
#define FIN freopen("input.txt","r",stdin)
#define FOUT freopen("output.txt","w+",stdout)using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
typedef pair<LL, int> PLI;const int MX = 1e5 + 5;
const int INF = 0x3f3f3f3f;int Head[MX], Next[MX], rear;struct Edge {int u, v, cost, a, b;
} E[MX];void edge_init() {rear = 0;memset(Head, -1, sizeof(Head));
}void edge_add(int u, int v, int cost, int a, int b) {E[rear].u = u;E[rear].v = v;E[rear].cost = cost;E[rear].a = a;E[rear].b = b;Next[rear] = Head[u];Head[u] = rear++;
}LL d[MX];
void dijistra(int Begin) {memset(d, INF, sizeof(d));d[Begin] = 0;priority_queue<PLI, vector<PLI>, greater<PLI> >work;work.push(PLI(0, Begin));while(!work.empty()) {PLI f = work.top();work.pop();LL dist = f.first;int u = f.second;for(int id = Head[u]; ~id; id = Next[id]) {int tq = E[id].a + E[id].b, time = dist % tq, add = 0;if(time > E[id].a || time + E[id].cost > E[id].a) add = tq - time;int cost = E[id].cost + add, v = E[id].v;if(dist + cost < d[v]) {d[v] = dist + cost;work.push(PLI(dist + cost, v));}}}
}int main() {int n, m, s, t, ansk = 0; //FIN;while(~scanf("%d%d%d%d", &n, &m, &s, &t)) {edge_init();for(int i = 1; i <= m; i++) {int u, v, a, b, cost;scanf("%d%d%d%d%d", &u, &v, &a, &b, &cost);if(a >= cost) edge_add(u, v, cost, a, b);}dijistra(s);printf("Case %d: %lld\n", ++ansk, d[t]);}return 0;
}