1.单源最短路问题1(Bellman-Ford算法)
//从顶点from指向顶点to的权值为cost的边 struct edge { int from, to, cost; }; edge es[max]; //边 int d[maxn]; //最短距离 int V, E; //V是顶点数,E是边数 //求解从顶点s出发到所有点的最短距离 void short_path(int s) { for (int i = 0; i < V; i++){ d[i] = INF; } d[s] = 0; while (true){ bool update = false; for (int i = 0; i < E; i++){ edge e = es[i]; if (d[e.from] != INF && d[e.to] > d[e.from] + e.cost){ d[e.to] = d[e.from] + e,cost; update = true; } } if (!update) break; } }
//如果返回true则存在负圈 bool find_negative_loop() { memset(d, 0, sizeof(d)); for (int i = 0; i < V; i++){ for (int j = 0; j < E; j++){ edge e = es[j]; if (d[e.to] > d[e.from] + e.cost){ d[e.to] = d[e.from] + e.cost; //如果第n次仍然更新了,则存在负圈 if (i == V - 1) return true; } } } return false; }
int cost[maxn][maxn]; //cost[u][v]表示边e=(u,v)的权值(不存在这条边时设为INF) int d[maxn]; //顶点s出发的最短距离 bool used[maxn]; //已经使用过的图 int V; //顶点数 //求从起点s出发到各个顶点的最短距离 void dijkstra(int s) { fill(d, d + V, INF); fill(used, used + V; false); d[s] = 0; while (true){ int v = -1; //从尚未使用过的顶点中选择一个距离最小的顶点 for (int u = 0; u < V; u++){ if (!used[u] && (v == -1 || d[u] < d[v])) v = u; } if (v == -1) break; used[v] = true; for (int u = 0; u < V; u++){ d[u] = min(d[u], d[v] + cost[v][u]); } } }
struct edge {int to, cost}; typedef pair<int, int> P; //first是最短距离,second是顶点的编号 int V; vector<edge> G[maxn]; int d[maxn]; void dijkstra(int s) { //通过指定greater<P>参数,堆按照first从小到大的顺序取出值 priority_queue<P, vector<P>, greater<P> > que; fill(d, d + V; INF); d[s] = 0; que.push(P(0, s)); while (!que.empty()){ P p = que.top(); que.pop(); int v = p.second; if (d[v] < p.first) continue; for (int i = 0; i < G[v].size(); i++){ edge e = G[v][i]; if (d[e.to] > d[v] + e.cost){ d[e.to] = d[v] + e.cost; que.push(P(d[e.to], e.to)); } } } }
int d[maxn][maxn]; //d[u][v]表示边e = (u,v)的权值(不存在时设为INF,不过d[i][i] = 0) int V; //顶点数 void warshall_floyd() { for (int k = 0; k < V; k++){ for (int i = 0; i < V; i++){ for (int j = 0; j < V; j++){ d[i][j] = min(d[i][j], d[i][k] + d[k][j]); } } } }
int prev[maxn]; //最短路上的前驱结点 //求从起点s出发到各个顶点的最短距离 void dijkstra(int s) { fill(d, d + V; INF); fill(used, used + V, false); fill(prev, prev + V, -1); d[s] = 0; while (true){ int v = -1; for (int u = 0; u < V; u++){ if (!used[u] && (v == -1 || d[u] < d[v])) v = u; } if (v == -1) break; used[v] = true; for (int u = 0; u < V; u++){ if (d[u] > d[v] + cost[v][u]){ d[u] = d[v] + cost[v][u]; prev[u] = v; } } } } //到顶点t的最短路 vector<int> get_path(int t) { vector<int> path; for ( ; t != -1; t = prev[t]) path.push_back(t); //不断沿着prev[t]走直到t = s; reverse(path.begin(), path.end()); return path; }