dijkstra算法

对于图有两种,一种是矩阵式的方方正正的,另一种是链表式的没有固定方向的,对于第一种

cpp 复制代码
int n;
std::vector<std::vector<int>>a(n, std::vector<int>(n)),dis(n,std::vector<int>(n));
int dx[5] = { 0,1,0,-1,0 };
int dy[5] = { 0,0,1,0,-1 };
using  PII= std::pair<int, int>;
PII st,end;
struct node {
	int x, y, dis;
	bool operator>(const node& other)const {
		return dis > other.dis;
	}
};
int dijkstra() {
	std::priority_queue<node>q;
	q.push({ st.first, st.second, 0 });
	dis[st.first][st.second] = 0;
	while (q.empty()) {
		node t = q.top();
		q.pop();
		if (t.x == end.first && t.y == end.second)return t.dis;
		for (int i = 1; i <=4; i++) {
			int nx = t.x + dx[i];
			int ny = t.y + dy[i];
			if (nx >= 0 && nx < n && ny >= 0 && ny < n) {
				int ndis = t.dis + a[nx][ny];
				if (ndis < dis[nx][ny]) {
					dis[nx][ny] = ndis;
					q.push({ nx,ny,ndis });
				}
			}
		}
	}
	return -1;
}
int main() {
	//输入获取起点终点和地图
	dijkstra();
	//输出结果
}

本质上是个广搜,和第二种相同的方面有以下几点,我们需要一个dis数组来存每个节点的搜索值,用于储存当前节点的最优值,本质框架不变,使用优先队列作为主要结构,但是这种判到终点的就是返回,且需要判断是否出界.还需要定义一个辅助方向数组.

来看看第二种

我们先来看有向图的情况

【模板】单源最短路径(标准版) - 洛谷

cpp 复制代码
constexpr int MaxM = 500010;
constexpr int MaxN = 100010;
struct edge {
	int to, dis, next;
};
edge e[MaxM];
int head[MaxN], dis[MaxN], cnt;
bool vis[MaxN];
int n, m, s;
inline void add_edge(int u, int v, int d) {
	cnt++;
	e[cnt].dis = d;
	e[cnt].to = v;
	e[cnt].next = head[u];
	head[u] = cnt;
}
struct node {
	int dis;
	int pos;
	bool operator<(const node& x)const {
		return x.dis < dis;
	}
};
std::priority_queue<node>q;
inline void dijkstra() {
	dis[s] = 0;
	q.push({ 0, s });
	while (!q.empty()) {
		node tmp = q.top();
		q.pop();
		int x = tmp.pos, d = tmp.dis;
		if (vis[x])continue;
		vis[x] = 1;
		for (int i = head[x]; i; i = e[i].next) {
			int y = e[i].to;
			if (dis[y] > dis[x] + e[i].dis) {
				dis[y] = dis[x] + e[i].dis;
				if (!vis[y]) {
					q.push({ dis[y],y });
				}
			}
		}
	}
}
int main() {
	std::cin >> n >> m >> s;
	for (int i = 1; i <= n; i++)dis[i] = 0x7fffffff;
	for (register int i = 0; i < m; i++) {
		register int u, v, d;
		std::cin >> u >> v >> d;
		add_edge(u, v, d);
	}
	dijkstra();
	for (int i = 1; i <= n; i++) {
		std::cout << dis[i]<<' ';
	}
	return 0;
}

有向图我们只需要正着存一次,并把dis数组初始化为INT_MAX.

而对于无向图,我们需要存两次,并把边数扩大为原来的两倍,相当于一去一回这个样子.

cpp 复制代码
constexpr int MAXM = 12410; // 修改为无向边转化后的边数的两倍
inline void add_edge(int u, int v, int d) {
	cnt++;
	// 第一条有向边:from u to v
	e[cnt].to = v;
	e[cnt].dis = d;
	e[cnt].next = h[u];
	h[u] = cnt;

	// 第二条有向边:from v to u
	cnt++;
	e[cnt].to = u;
	e[cnt].dis = d;
	e[cnt].next = h[v];
	h[v] = cnt;
}
相关推荐
夜天炫安全9 分钟前
数据结构中所需的C语言基础
c语言·数据结构·算法
2301_789015621 小时前
DS进阶:AVL树
开发语言·数据结构·c++·算法
qyzm4 小时前
天梯赛练习(3月13日)
开发语言·数据结构·python·算法·贪心算法
逆境不可逃5 小时前
LeetCode 热题 100 之 64. 最小路径和 5. 最长回文子串 1143. 最长公共子序列 72. 编辑距离
算法·leetcode·动态规划
CoderCodingNo5 小时前
【GESP】C++五级练习题 luogu-P1182 数列分段 Section II
开发语言·c++·算法
放下华子我只抽RuiKe55 小时前
机器学习全景指南-直觉篇——基于距离的 K-近邻 (KNN) 算法
人工智能·gpt·算法·机器学习·语言模型·chatgpt·ai编程
kisshuan123965 小时前
[特殊字符]【深度学习】DA3METRIC-LARGE单目深度估计算法详解
人工智能·深度学习·算法
sali-tec5 小时前
C# 基于OpenCv的视觉工作流-章33-Blod分析
图像处理·人工智能·opencv·算法·计算机视觉
Eward-an6 小时前
LeetCode 239. 滑动窗口最大值(详细技术解析)
python·算法·leetcode
一叶落4386 小时前
LeetCode 50. Pow(x, n)(快速幂详解 | C语言实现)
c语言·算法·leetcode