整体上一般分为 有向图 和 无向图。
加权有向图,就是图中边是有权值的,加权无向图也是同理。
无向图中有几条边连接该节点,该节点就有几度
在有向图中,每个节点有出度和入度。出度:从该节点出发的边的个数。入度:指向该节点边的个数。
在图中表示节点的连通情况,我们称之为连通性
一般使用邻接表、邻接矩阵 或者用类来表示。主要是 朴素存储、邻接表和邻接矩阵。
邻接矩阵 使用 二维数组来表示图结构。 邻接矩阵是从节点的角度来表示图,有多少节点就申请多大的二维数组。
优点:
缺点:
优点:
缺点:
关键就两点:
代码框架
void dfs(参数) {
if (终止条件) {
存放结果;
return;
}
for (选择:本节点所连接的其他节点) {
处理节点;
dfs(图,选择的节点); // 递归
回溯,撤销处理结果
}
}
def dfs(graph, x, n, result, path):
# 当前遍历的节点x 到达节点n
if x == n: # 找到符合条件的一条路径
result.append(path[:])
return
for i in range(1, n+1): # 遍历节点x链接的所有节点
if graph[x][i] == 1:
path.append(i)
dfs(graph, i, n, result, path)
path.pop()
if __name__ == "__main__":
n, m = map(int, input().strip().split())
# 节点编号从1到n,所以申请 n+1 这么大的数组
graph = [[0] * (n + 1) for _ in range(n+1)]
for _ in range(m):
s, t = map(int, input().strip().split())
# 使用邻接矩阵 表示无线图,1 表示 s 与 t 是相连的
graph[s][t] = 1
result = []
dfs(graph, 1, n, result, [1])
# 输出结果
if len(result) == 0:
print(-1)
for path in result:
print(" ".join([str(i) for i in path]))
def dfs(graph, x, n, result, path):
# 当前遍历的节点x 到达节点n
if x == n: # 找到符合条件的一条路径
result.append(path[:])
return
for i in graph[x]: # 遍历节点x链接的所有节点
path.append(i)
dfs(graph, i, n, result, path)
path.pop()
if __name__ == "__main__":
n, m = map(int, input().strip().split())
# 节点编号从1到n,所以申请 n+1 这么大的数组
graph = [[] for _ in range(n+1)] # 邻接表
for _ in range(m):
s, t = map(int, input().strip().split())
# 使用邻接表
graph[s].append(t)
result = []
dfs(graph, 1, n, result, [1])
# 输出结果
if len(result) == 0:
print(-1)
for path in result:
print(" ".join([str(i) for i in path]))
主要在生成图和遍历图的时候不一样
广搜的搜索方式就适合于解决两个点之间的最短路径问题。因为广搜是从起点出发,以起始点为中心一圈一圈进行搜索,一旦遇到终点,记录之前走过的节点就是一条最短路。
int dir[4][2] = {0, 1, 1, 0, -1, 0, 0, -1}; // 表示四个方向
// grid 是地图,也就是一个二维数组
// visited标记访问过的节点,不要重复访问
// x,y 表示开始搜索节点的下标
void bfs(vector>& grid, vector>& visited, int x, int y) {
queue> que; // 定义队列
que.push({x, y}); // 起始节点加入队列
visited[x][y] = true; // 只要加入队列,立刻标记为访问过的节点
while(!que.empty()) { // 开始遍历队列里的元素
pair cur = que.front(); que.pop(); // 从队列取元素
int curx = cur.first;
int cury = cur.second; // 当前节点坐标
for (int i = 0; i < 4; i++) { // 开始想当前节点的四个方向左右上下去遍历
int nextx = curx + dir[i][0];
int nexty = cury + dir[i][1]; // 获取周边四个方向的坐标
if (nextx < 0 || nextx >= grid.size() || nexty < 0 || nexty >= grid[0].size()) continue; // 坐标越界了,直接跳过
if (!visited[nextx][nexty]) { // 如果节点没被访问过
que.push({nextx, nexty}); // 队列添加该节点为下一轮要遍历的节点
visited[nextx][nexty] = true; // 只要加入队列立刻标记,避免重复访问
}
}
}
}