7-35 城市间紧急救援 (25 分)

这道题已经做了很多遍了,但是这次做的时候还是出现了一个bug
主要是没有注意scanf的输入特性,读入边的时候,不能同时处理边读入边使用的行为。
必须先读完然后再使用。

#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
const int N = 505;
const int INF = 0x3f3f3f3f;
int m, n, s, D, cntPath = 0, optN = 0;

int G[N][N];
int c[N], d[N], vis[N];
vector<int> pre[N], path, optPath;
void Dijkstra(){
	memset(d, 0x3f, sizeof(d)); //最短距离置为无穷大 
	memset(vis, 0, sizeof(vis));
	d[s] = 0;
	for(int i = 0; i < n; i++){ //循环N次 
		int t = -1;
		for(int j = 0; j < n; j++){
			if(!vis[j] &&(t==-1 || d[t] > d[j])){
				t = j;
			}
		}
		if(t == -1) return;
		vis[t] = 1;
		for(int j = 0; j < n; j++){
			if(!vis[j]){
				if(d[t] + G[t][j] < d[j]){
					pre[j].clear(); 
					d[j] = d[t] + G[t][j];
					pre[j].push_back(t);
				}else if(d[t] + G[t][j] == d[j]){
					pre[j].push_back(t);
				} 
			}
		}
	} 
}
void dfs(int u){
	if(u == s){
		cntPath++;
		path.push_back(u);
		int tmpN = 0;
		for(int j = 0; j < path.size(); j++){
			int v = path[j];
			tmpN += c[v];
		}
		if(tmpN > optN){
			optN = tmpN;
			optPath = path;
		}
		path.pop_back();
		return;
	}
	path.push_back(u);
	for(int j = 0; j < pre[u].size(); j++){
		int v = pre[u][j];
		dfs(v);
	}
	path.pop_back();
}
int main() {
	int u, v, w;
	memset(G, 0x3f, sizeof G);
	scanf("%d%d%d%d", &n, &m, &s, &D);
	for(int i = 0; i < n; i++){
		scanf("%d", &c[i]);
	}
	
	for(int i = 0; i < m; i++){
		scanf("%d%d%d", &u, &v, &w); //读入边的时候需要注意 
		G[v][u] = G[u][v] = w; //是无向图 
	}
	Dijkstra();
	dfs(D);
	printf("%d %d\n", cntPath, optN);
	for(int j = optPath.size()-1; j >= 0 ; j--){
		printf("%d", optPath[j]);
		if(j) printf(" ");
	}
	
	return 0;
}

你可能感兴趣的:(数据结构与算法题目集,最短路径)