PAT甲级刷题实录——1003

原题链接

https://pintia.cn/problem-sets/994805342720868352/problems/994805523835109376

思路

这题基本上就是采用迪杰斯特拉算法求最短路径。只是我一开始给想复杂化了,结果走了不少弯路。因为城市的数量最多可以 500 个,是个相当大的数,同时并不是每个城市之间都有路,如果用矩阵存储图的话是个稀疏矩阵,因此我就想用邻接表存。这下可好,光是读入数据到邻接表就写了我 80 行代码,之后的算法更是无从下手了,完全走进死胡同了,无奈之下,只好去网上查人家的做法,发现人家压根就没考虑稀疏矩阵的事,都是用邻接矩阵存,更有甚者直接定义了一个 500*500 的矩阵,空间复杂度高到了极致。我还是折中一下吧,用题目中给的城市总数定义邻接矩阵大小,实现方法还是我们最喜欢的 vector,为此我特意搜索了一下 vector 实现二维数组的方法,应该是这样的 vector<vector<int> > matrix(n, vector<int>(m)) ,注意两个 > 之间有空格。这道题需要在传统迪杰斯特拉算法上增加的东西是:1. 权值相同的最短路径总数。2. 可以集合到的救援队总数。这些在代码里都有体现,应该一看就能明白。

代码

#include <iostream>
#include <vector>
#include <limits.h>
using namespace std;

int main()
{
	int cityNum, roadNum, depa, dest;
	cin >> cityNum >> roadNum >> depa >> dest;
	vector<vector<int> > adjMatrix(cityNum, vector<int>(cityNum,INT_MAX));	//邻接矩阵
	vector<int> dist(cityNum,INT_MAX);	//记录到各个城市的最短距离
	vector<int> pathSum(cityNum);	//到各个城市的最短路径数量
	vector<int> teamSum(cityNum);	//到各个城市能集合到的救援队数量
	vector<bool> reach(cityNum);	//记录计算过程中是否以最短路径到各个城市
	vector<int> teams(cityNum);	//每个城市救援队数量
	for (int i = 0; i < cityNum; i++)
	{
		adjMatrix[i][i] = 0;
	}
	for (int i = 0; i < cityNum; i++)
	{
		int tnum;
		cin >> tnum;
		teams[i]=tnum;
	}
	for (int i = 0; i < roadNum; i++)
	{
		int from, to, weight;
		cin >> from >> to >> weight;
		adjMatrix[from][to] = weight;
		adjMatrix[to][from] = weight;
		if (from == depa)	//根据出发点的相邻路径更新相关信息
		{
			dist[to] = weight;
			teamSum[to] = teams[from] + teams[to];
			pathSum[to] = 1;
		}
	}
	dist[depa] = 0;
	teamSum[depa] = teams[depa];
	pathSum[depa] = 1;
	reach[depa] = true;
	//注意,只有当reach[i]为true时,dist[i]才是出发点到它真正的最短路径长度,否则就还得更新
	for (int i = 0; i < cityNum; i++)
	{
		int min = INT_MAX, u=-1;
		for (int j = 0; j < cityNum; j++)
		{
			if (reach[j] == false && dist[j] < min)
			{
				min = dist[j];
				u = j;
			}
		}
		if (min == INT_MAX)
			break;
		reach[u] = true;
		for (int j = 0; j < cityNum; j++)
		{
			if (reach[j] == false && adjMatrix[u][j]!=INT_MAX && dist[u] + adjMatrix[u][j] < dist[j])	
				//若adjMatrix[u][j]==INT_MAX就直接排除掉,如果没有判断,那么当它为INT_MAX的时候,dist[u]+adjMatrix[u][j]会反转变成负数,这样反而会符合<dist[j]的要求
			{
				dist[j] = dist[u] + adjMatrix[u][j];	//更新最短距离
				pathSum[j] = pathSum[u];	//更新可达的最短路径总数
				teamSum[j] = teamSum[u] + teams[j];		//更新可以集合到的救援队总数
			}
			else if (reach[j] == false && (dist[u] + adjMatrix[u][j]) == dist[j])	//如果有另一条最短距离相同的路径
			{
				pathSum[j] += pathSum[u];	//增加可达的最短路径总数
				if (teamSum[j] < teamSum[u] + teams[j])	//如果这条路径可以集合到的救援队数量更多
					teamSum[j] = teamSum[u] + teams[j];	//更新可以集合到的救援队数量
			}
		}
	}
	cout << pathSum[dest] << ' ' << teamSum[dest];
}

感想和注意事项

  1. 当节省空间的方法过于复杂时,尽量用空间换时间,而不是想着怎么去节省空间。比如说图用邻接矩阵而不是邻接表存,这样读入数据以及之后的运算过程都会快很多。
  2. 代码中用到了一个 INT_MAX 值,表示不可达的路径长度,这个常量包含在 limits.h 头文件中。但是需要注意的是,如果把 INT_MAX 的值加上一个正数以后,会反转变成负数,从而导致程序出现错误。因此我在代码中进行了判断。我看网上有些方法就是自己定义一个常量 INF,再给它手工赋一个很大但又不超过 int 表示范围的数比如 1111111,从而来表示不可达的路径长度,这样也行,但我个人感觉不太严谨。
  3. vector 真是神器,太好用了。当然你也可以直接定义一个静态数组,用一个很大的数表示它的容量,事实上网上大部分解答都是这么做的,但我不用这种方法也是和上面一样的原因,这种做法太不严谨而且太浪费空间了。当能够用很低的代价(多几行代码)节省大量的空间时,这种做法还是值得的。
posted @ 2020-01-10 21:54  aopstudio  阅读(167)  评论(0编辑  收藏  举报
Live2D