dijkstra+链式前向星+优先队列优化

Invitation Cards POJ - 1511

Description
n-1个人从1号点出发,到剩余n-1个宣传点,然后再回到1号点汇报结果,求所有人往返路径和的最小值

Input
输入由T个案例组成。输入的第一行只包含正整数T。
接下来是N和M,1 <= N,M <= 1000000,表示N个点和连接N个点的M条边。
然后有M行,每行包括三个值U,V,W,表示从点U到点V需要W的路程。你可以假设该图连通。
注意是单向通道!!!

Output
对于每个案例,打印一行,表示路径总和的最小值。

Sample Input
2
2 2
1 2 13
2 1 33
4 6
1 2 10
2 1 60
1 3 20
3 4 10
2 4 5
4 1 50

Sample Output
46
210

**思路:**题目让我们求往返最短路的最小值,可以用dijkstra正向和反向都求一次,相加即可。
**注意:**题目数据范围太大了,用邻接矩阵存图肯定会T,所以我们选择用链式前向星存图(才发现链式前向星这么猛)但没想到的是- -,dijkstra没有用优先队列优化还是会T,草率了,后面用了优先队列+链式前向星过的- -
ac代码如下:

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cctype>
#include<iomanip>
#include<map>
#include<vector>
#include<list>
#include<deque>
#include<stack>
#include<queue>
#include<set>
#include<cctype>
#include<string>
#include<stdexcept>
#include<fstream>
#include<sstream>
#define mem(a,b) memset(a,b,sizeof(a))
#define mod 10000007
#define debug() puts("what the fuck!")
#define dedebug() puts("what the fuck!!!")
#define ll long long
#define ull unsigned long long
#define speed {ios::sync_with_stdio(false); cin.tie(0); cout.tie(0); };
using namespace std;
const double PI = acos(-1.0);
const int maxn = 1e6 + 50;
const int N = 35;
const int INF = 0x3f3f3f3f;
const int inf = 0xfffffff;//比INF小,防止累加爆int
const double esp_0 = 1e-6;
const double gold = (1 + sqrt(5)) / 2;
int gcd(int x, int y) {
	return y ? gcd(y, x % y) : x;
}
int n, m;
int flag;
struct node {
	int next, to, w;
}edge[maxn],redge[maxn];
int head[maxn], rhead[maxn], cnt, rcnt;
int dis[maxn], rdis[maxn], vis[maxn];
void add(int u, int v, int w) {
	edge[cnt].to = v;
	edge[cnt].next = head[u];
	edge[cnt].w = w;
	head[u] = cnt++;
}
void radd(int u, int v, int w) {
	redge[rcnt].to = v;
	redge[rcnt].next = rhead[u];
	redge[rcnt].w = w;
	rhead[u] = rcnt++;
}
void init() {
	mem(dis, INF);
	mem(rdis, INF);
	mem(head, -1);
	mem(rhead, -1);
}
struct que {
	int id, dist;
	que() {};
	que(int id, int dist) :id(id), dist(dist) {};
	bool operator<(const que& a)const{
		return dist > a.dist;
	}

};
void dijkstra(int dis[],int head[],node edge[]) {
	mem(vis, 0);
	dis[1] = 0;
	priority_queue<que>q;
	q.push(que(1, dis[1]));
	while (!q.empty()) {
		que step = q.top();
		q.pop();
		if (vis[step.id])continue;
		vis[step.id] = 1;
		for (int i = head[step.id]; i != -1; i = edge[i].next) {
			int value = edge[i].to;
			if (!vis[value] && dis[value] > dis[step.id] + edge[i].w) {
				dis[value] = dis[step.id] + edge[i].w;
				q.push(que(value, dis[value]));
			}
		}
	}
}
int main() {
	int t;
	scanf("%d", &t);
	while (t--) {
		init();
		scanf("%d%d", &n, &m);
		for (int i = 1; i <= m; ++i) {
			int u, v, w;
			scanf("%d%d%d", &u, &v, &w);
			add(u, v, w);
			radd(v, u, w);
		}
		dijkstra(dis,head,edge);
		dijkstra(rdis, rhead,redge);
		ll ans = 0;
		for (int i = 1; i <= n; ++i) {
//			printf("%d %d\n", dis[i], rdis[i]);
			ans += dis[i] + rdis[i];
		}
		printf("%lld\n", ans);
	}
	return 0;
}

未用优先队列优化的错误代码

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cctype>
#include<iomanip>
#include<map>
#include<vector>
#include<list>
#include<deque>
#include<stack>
#include<queue>
#include<set>
#include<cctype>
#include<string>
#include<stdexcept>
#include<fstream>
#include<sstream>
#define mem(a,b) memset(a,b,sizeof(a))
#define mod 10000007
#define debug() puts("what the fuck!")
#define dedebug() puts("what the fuck!!!")
#define ll long long
#define ull unsigned long long
#define speed {ios::sync_with_stdio(false); cin.tie(0); cout.tie(0); };
using namespace std;
const double PI = acos(-1.0);
const int maxn = 1e6 + 50;
const int N = 35;
const int INF = 0x3f3f3f3f;
const int inf = 0xfffffff;//比INF小,防止累加爆int
const double esp_0 = 1e-6;
const double gold = (1 + sqrt(5)) / 2;
int gcd(int x, int y) {
	return y ? gcd(y, x % y) : x;
}
int n, m;
struct node {
	int next, to, w;
}edge[maxn],redge[maxn];
int head[maxn], rhead[maxn], cnt, rcnt;
int dis[maxn], rdis[maxn], vis[maxn];
void add(int u, int v, int w) {
	edge[cnt].to = v;
	edge[cnt].next = head[u];
	edge[cnt].w = w;
	head[u] = cnt++;
}
void radd(int u, int v, int w) {
	redge[rcnt].to = v;
	redge[rcnt].next = rhead[u];
	redge[rcnt].w = w;
	rhead[u] = rcnt++;
}
void init() {
	mem(dis, INF);
	mem(rdis, INF);
	mem(head, -1);
	mem(rhead, -1);
}
void dijkstra(int dis[],int head[],node edge[]) {
	mem(vis, 0);
	dis[1] = 0;
	for (int i = 1; i <= n; ++i) {
		int min = INF;
		int step = -1;
		for (int j = 1; j <= n; ++j) {
			if (!vis[j] && min > dis[j]) {
				min = dis[j];
				step = j;
			}
		}
		if (step == -1)break;
		vis[step] = 1;
		for (int i = head[step]; i != -1; i = edge[i].next) {
			int now = edge[i].to;
			if (dis[now] > dis[step] + edge[i].w) {
				dis[now] = dis[step] + edge[i].w;
			}
		}
	}
}
int main() {
	int t;
	scanf("%d", &t);
	while (t--) {
		init();
		scanf("%d%d", &n, &m);
		for (int i = 1; i <= m; ++i) {
			int u, v, w;
			scanf("%d%d%d", &u, &v, &w);
			add(u, v, w);
			radd(v, u, w);
		}
		dijkstra(dis,head,edge);
		dijkstra(rdis, rhead,redge);
		ll ans = 0;
		for (int i = 1; i <= n; ++i) {
//			printf("%d %d\n", dis[i], rdis[i]);
			ans += dis[i] + rdis[i];
		}
		printf("%lld\n", ans);
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在使用Dijkstra算法求解最短路径时,可以使用链式前向星(Linked List Representation)来表示图的数据结构,以提高算法的效率。 链式前向星是一种用于存储稀疏图的数据结构,它将每个节点的邻接边以链表的形式存储。具体实现步骤如下: 1. 创建一个结构体或类来表示图的边,包括目标节点和边的权重等信息。 2. 创建一个数组,数组的每个元素表示一个节点,每个节点包含一个指向邻接边链表的指针。 3. 遍历图的所有边,对于每条边(u, v)和权重w,创建一个新的边节点,并将其插入到节点u的邻接边链表中。 4. 在Dijkstra算法中,需要使用一个优先队列(最小堆)来选择最短路径的节点。队列中的元素包括节点索引和到达该节点的距离。 5. 初始化距离数组和标记数组,距离数组记录起始节点到各个节点的最短距离,标记数组用于标记已经找到最短路径的节点。 6. 将起始节点加入优先队列,并将距离数组中起始节点的距离设为0。 7. 重复以下步骤,直到优先队列为空: a. 从优先队列中取出距离最小的节点u。 b. 遍历节点u的邻接边链表,对于每个邻接节点v,如果通过u到达v的距离更短,则更新距离数组中节点v的距离,并将节点v加入优先队列。 8. 当所有节点都被标记后,最短路径的结果就可以通过距离数组得到。 使用链式前向星可以减少遍历边的次数,提高Dijkstra算法的效率。同时,链式前向星还可以支持动态图的操作,如添加和删除边。但是,它需要额外的空间来存储链表和边节点,因此适用于稀疏图而非稠密图。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值