最短路径-Dijkstra

7-6 最短路径-Dijkstra

分数 15

全屏浏览题目

切换布局

作者 唐艳琴

单位 中国人民解放军陆军工程大学

城市的道路四通八达,我们经常需要查找从某地出发到其他地方的路径,当然我们希望能最快到达。现得到去每个地方需要花费的时间,现请你编写程序,计算从特定地点出发到所有城市之间的最短时间。

输入格式:

输入的第一行给出城市数目N (1≤N≤10)和道路数目M和1(表示有向图)或0(表示无向图);

接下来的M行对应每个城市间来往所需时间,每行给出3个正整数,分别是两个城市的编号(从1编号到N)和来往两城市间所需时间。最后一行给出一个编号,表示从此编号地点出发。

输出格式:

输出从特定地点出发到达所有城市(按编号1-编号N顺序输出)的距离(用编号1->编号**: 表示 ),如果无路,请输出no path。每个城市占一行。

输入样例:

4 4 1
1 2 2
1 4 8
3 2 16
3 4 10
1

输出样例:

1->1:0
1->2:2
1->3:no path
1->4:8

代码如下

#include<bits/stdc++.h>

#include<queue>

using namespace std;

#define N 100006

int ver[N * 2];

int dist[N * 2], edge[N * 2], head[N * 2], nex[N * 2], v[N * 2];





//有向边(x,y),权值为z

int tot = 0;

void add(int x, int y, int z)

{

	ver[++tot] = y, edge[tot] = z;

	nex[tot] = head[x], head[x] = tot;//head,nex构成类似链表

}

typedef pair<int, int>PII;

priority_queue<PII, vector<PII>, greater<PII>>q;//小顶堆

void dijkstra(int s)

{

	memset(dist, 0x3f, sizeof dist);

	memset(v, 0, sizeof v);

	dist[s] = 0;//出发点

	q.push({ 0,s });

	int x;



	while (q.size())

	{

		x = q.top().second;

		q.pop();

		if (v[x])continue;//已经经过该点

		v[x] = 1;

		for (int i = head[x]; i; i = nex[i])

		{

			int y = ver[i];

			int z = edge[i];

			if (dist[y] > dist[x] + z)

			{

				dist[y] = dist[x] + z;

				q.push({ dist[y],y });



			}



		}



	}

}

int main()

{

	int n, m;

	cin >> n >> m;

	int a;

	cin >> a;

	int x, y, z;

	for (int i = 0; i < m; i++)

	{

		cin >> x >> y >> z;

		if (a == 1)//有向图

			add(x, y, z);

		else if (a == 0) {//无向图

			add(x, y, z);

			add(y, x, z);
		}//双向加上

	}

	int start;

	cin >> start;

	dijkstra(start);

	for (int i = 1; i <= n; i++)

	{
		if (dist[i] < 0x3f)//

			cout << start << "->" << i << ":" << dist[i] << endl;

		else cout << start << "->" << i << ":" << "no path" << endl;//没有这一路径时的判断条件是dist的值为初始化的值,但无穷大的数值可能不一定


	}

}

  • 12
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值