继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

广度优先入门:初学者必读指南

Qyouu
关注TA
已关注
手记 489
粉丝 88
获赞 414
概述

广度优先搜索(Breadth-First Search,BFS)是一种在树形结构或图结构中遍历节点的算法。它从树的根节点或图中的任意一个顶点开始,逐层向外围进行遍历,每次先遍历完当前层的所有节点后,再向下一层进行遍历。

广度优先搜索简介

广度优先搜索是一种层次遍历算法,它首先遍历起始节点,然后遍历起始节点的直接邻居节点,再遍历这些直接邻居节点的邻居节点,以此类推,直到遍历完所有可达节点。这种遍历方法的一个重要特性是它总是首先访问离起始节点最近的节点,因此在寻找最短路径的问题中特别有用。

广度优先搜索通常使用队列来实现,每一步遍历中,当前节点会从队列中移出,并将其未访问过的邻居节点加入队列。这样,队列中的节点始终表示了尚未完全遍历的节点集合。

广度优先搜索在许多领域都有应用,例如在游戏开发中,广度优先搜索可以用来计算玩家角色和敌人角色之间的最短路径;在图论中,广度优先搜索可以用来检测图中的连通性;在社交网络中,广度优先搜索可以用来寻找用户之间的最短联系链。

广度优先搜索在以下场景中特别有用:

  1. 最短路径问题:广度优先搜索可以在无权图中找到两个节点之间的最短路径,即最短路径是从起点到终点经过的边数目最少的路径。
  2. 图的连通性检测:广度优先搜索可以用来检测图是否连通,即从一个节点出发,是否能够遍历到图中的所有节点。
  3. 搜索树的生成:广度优先搜索可以生成搜索树,这种搜索树的生成方式可以方便地找到路径或者验证节点是否可达。
  4. 社交媒体网络分析:在社交网络中,广度优先搜索可以用来寻找两个用户之间的最短联系路径,从而可以分析用户在网络中的位置和影响力。
广度优先搜索算法详解

广度优先搜索的步骤

广度优先搜索的基本步骤如下:

  1. 初始化:将起始节点放入队列中。
  2. 遍历:从队列中取出一个节点,检查该节点的相邻节点。
  3. 标记与检查:将这些相邻节点标记为已访问,并将未访问过的相邻节点加入队列。
  4. 重复:重复步骤2和3,直到队列为空,即遍历完所有可达节点。

广度优先搜索的实现思路

广度优先搜索的实现思路通常遵循以下步骤:

  1. 初始化队列:将起始节点放入队列中。
  2. 遍历节点:使用一个循环来遍历队列中的节点,并在每次循环中执行以下操作:
    • 从队列中取出一个节点。
    • 检查该节点的相邻节点,确保这些相邻节点未被访问过。
    • 将这些相邻节点标记为已访问,并将它们加入队列。
  3. 标记已访问:在访问每个节点时,用一个集合标记这些节点已被访问。
  4. 检查队列为空:当队列为空时,遍历结束。

从理论上讲,广度优先搜索的时间复杂度为 O(V + E),其中 V 是图中的顶点数量,E 是图中的边数量。这种线性时间复杂度使得广度优先搜索在处理大规模图数据时具有较高的效率。

广度优先搜索实现代码示例

Python实现广度优先搜索代码

下面是一个简单的广度优先搜索实现,用于遍历一个无向图。这个实现使用了一个字典来表示图,其中每个键值对表示一个节点及其相邻节点。为了标记已访问的节点,使用了一个集合。

def bfs(graph, start):
    visited = set()
    queue = [start]

    while queue:
        node = queue.pop(0)
        if node not in visited:
            print(node, end=" ")
            visited.add(node)
            queue.extend(graph[node] - visited)

# 示例图:{节点: 相邻节点集合}
graph = {
    'A': {'B', 'C'},
    'B': {'A', 'D', 'E'},
    'C': {'A', 'F'},
    'D': {'B'},
    'E': {'B', 'F'},
    'F': {'C', 'E'}
}

bfs(graph, 'A')  # 输出: A B C D E F

Java实现广度优先搜索代码

下面是一个简单的广度优先搜索实现,用于遍历一个无向图。这个实现使用了一个邻接矩阵来表示图,并使用了一个集合来标记已访问的节点。

import java.util.*;

public class BFSExample {
    private static void bfs(int[][] graph, int start) {
        boolean[] visited = new boolean[graph.length];
        Queue<Integer> queue = new LinkedList<>();
        queue.offer(start);
        visited[start] = true;

        while (!queue.isEmpty()) {
            int node = queue.poll();
            System.out.print(node + " ");
            for (int i = 0; i < graph[node].length; i++) {
                if (graph[node][i] == 1 && !visited[i]) {
                    queue.offer(i);
                    visited[i] = true;
                }
            }
        }
    }

    public static void main(String[] args) {
        int[][] graph = {
            {0, 1, 1, 0, 0, 0},
            {1, 0, 1, 1, 0, 0},
            {1, 1, 0, 0, 1, 1},
            {0, 1, 0, 0, 0, 0},
            {0, 0, 1, 0, 0, 1},
            {0, 0, 1, 0, 1, 0}
        };

        bfs(graph, 0);  // 输出: 0 1 2 3 4 5
    }
}
广度优先搜索的优化技巧

如何提升广度优先搜索的效率

  1. 使用更高效的队列实现:在广度优先搜索中,队列的插入和删除操作是频繁发生的。使用更高效的队列实现可以提高程序的执行效率。例如,在Python中可以使用collections.deque来代替列表实现更高效的队列操作。
  2. 避免不必要的遍历:在处理大规模图数据时,避免不必要的遍历是非常重要的。可以通过提前检查节点是否已访问来避免重复遍历。
  3. 剪枝:如果在遍历过程中发现某些路径不可能到达目标,可以提前终止这些路径的遍历,从而减少不必要的计算。
  4. 代码示例:以下是一个使用collections.deque的示例代码,展示了如何提高队列操作的效率。
from collections import deque

def bfs(graph, start):
    visited = set()
    queue = deque([start])

    while queue:
        node = queue.popleft()
        if node not in visited:
            print(node, end=" ")
            visited.add(node)
            queue.extend(graph[node] - visited)

# 示例图:{节点: 相邻节点集合}
graph = {
    'A': {'B', 'C'},
    'B': {'A', 'D', 'E'},
    'C': {'A', 'F'},
    'D': {'B'},
    'E': {'B', 'F'},
    'F': {'C', 'E'}
}

bfs(graph, 'A')  # 输出: A B C D E F

广度优先搜索中的常见问题及解决方法

  1. 节点重复访问:为了避免节点重复访问,需要使用一个集合来记录已访问的节点。在访问一个节点之前,先检查该节点是否已被访问过。
  2. 图中存在环:在遍历图时,如果图中存在环,则可能会导致无限循环。解决这个问题的方法是使用一个集合来标记已访问的节点。
  3. 内存使用问题:在处理大规模图数据时,可能会遇到内存使用问题。可以通过优化数据结构或使用更高效的算法来减少内存使用。
  4. 代码示例:以下是一个示例代码,展示了如何避免节点重复访问。
def bfs(graph, start):
    visited = set()
    queue = deque([start])

    while queue:
        node = queue.popleft()
        if node not in visited:
            print(node, end=" ")
            visited.add(node)
            queue.extend(graph[node] - visited)

# 示例图:{节点: 相邻节点集合}
graph = {
    'A': {'B', 'C'},
    'B': {'A', 'D', 'E'},
    'C': {'A', 'F'},
    'D': {'B'},
    'E': {'B', 'F'},
    'F': {'C', 'E'}
}

bfs(graph, 'A')  # 输出: A B C D E F
广度优先搜索练习题

初级难度的广度优先搜索题目

题目描述

给定一个无向图,找到从起点到终点的最短路径。

输入

  • 输入包含多组测试数据,第一行为测试数据的组数T。
  • 每组测试数据的第一行为三个整数N、M和S,分别表示节点的数量、边的数量和起点。
  • 接下来的M行,每行两个整数,表示一条边。
  • 最后一行给出终点。

输出

  • 对于每组测试数据,输出从起点到终点的最短路径长度,如果不存在路径则输出-1。

示例:

输入:
2
4 4 1
1 2
2 3
3 4
4 1
4
5 5 1
1 2
2 3
3 4
4 5
5 1
5

输出:
4
5

解决方案

from collections import deque

def bfs_shortest_path(graph, start, end):
    queue = deque([(start, [start])])
    visited = set([start])

    while queue:
        node, path = queue.popleft()
        if node == end:
            return len(path) - 1
        for neighbor in graph[node]:
            if neighbor not in visited:
                queue.append((neighbor, path + [neighbor]))
                visited.add(neighbor)
    return -1

def solve_bfs_shortest_path():
    test_cases = int(input())
    results = []

    for _ in range(test_cases):
        N, M, S = map(int, input().split())
        graph = {i: set() for i in range(1, N + 1)}
        for _ in range(M):
            u, v = map(int, input().split())
            graph[u].add(v)
            graph[v].add(u)
        end = int(input())
        results.append(bfs_shortest_path(graph, S, end))

    for result in results:
        print(result)

solve_bfs_shortest_path()

中级难度的广度优先搜索题目

题目描述

给定一个图,找到从起点到终点的所有简单路径,简单路径是指没有重复节点的路径。

输入

  • 输入包含多组测试数据,第一行为测试数据的组数T。
  • 每组测试数据的第一行为三个整数N、M和S,分别表示节点的数量、边的数量和起点。
  • 接下来的M行,每行两个整数,表示一条边。
  • 最后一行给出终点。

输出

  • 对于每组测试数据,输出从起点到终点的所有简单路径,每条路径用一个列表表示。

示例:

输入:
2
4 4 1
1 2
2 3
3 4
4 1
4
5 5 1
1 2
2 3
3 4
4 5
5 1
5

输出:
[[1, 2, 3, 4]]
[[1, 2, 3, 4, 5]]

解决方案

from collections import deque

def bfs_all_paths(graph, start, end):
    queue = deque([(start, [start])])
    visited = set([start])
    paths = []

    while queue:
        node, path = queue.popleft()
        if node == end:
            paths.append(path)
        else:
            for neighbor in graph[node]:
                if neighbor not in path:
                    queue.append((neighbor, path + [neighbor]))
                    visited.add(neighbor)
    return paths

def solve_bfs_all_paths():
    test_cases = int(input())
    results = []

    for _ in range(test_cases):
        N, M, S = map(int, input().split())
        graph = {i: set() for i in range(1, N + 1)}
        for _ in range(M):
            u, v = map(int, input().split())
            graph[u].add(v)
            graph[v].add(u)
        end = int(input())
        results.append(bfs_all_paths(graph, S, end))

    for result in results:
        for path in result:
            print(path)

solve_bfs_all_paths()
广度优先搜索与其他搜索算法的比较

广度优先搜索与深度优先搜索的区别

广度优先搜索(BFS)和深度优先搜索(DFS)都是图的遍历算法,但它们在实现方式和应用场景上有明显的区别:

  • 遍历顺序
    • 广度优先搜索从起点开始,逐层向外遍历,每次遍历完当前层的所有节点,再向下一层进行遍历。
    • 深度优先搜索从起点开始,尽可能深入地遍历,直到无法深入为止,然后再回溯。
  • 实现方式
    • 广度优先搜索使用队列来记录待遍历的节点,队列中的节点始终表示了尚未遍历的节点集合。
    • 深度优先搜索使用栈或递归来记录待遍历的节点,栈中的节点或递归的调用栈表示了尚未回溯的节点集合。
  • 应用场景
    • 广度优先搜索通常用于寻找最短路径,适用于无权图。
    • 深度优先搜索适用于需要深入探索的问题,例如查找包含特定值的路径,或检测图中的循环。

广度优先搜索与启发式搜索的区别

广度优先搜索和启发式搜索(例如A*算法)都是图搜索算法,但它们在搜索策略上有明显的区别:

  • 搜索策略
    • 广度优先搜索是一种无策略搜索算法,它按照层次顺序遍历图中的节点。
    • 启发式搜索是一种基于启发信息的搜索算法,它使用一个评估函数来估计从当前节点到目标节点的最优路径,优先选择具有最小启发值的节点进行扩展。
  • 应用场景
    • 广度优先搜索适用于寻找最短路径或在无权图中进行遍历。
    • 启发式搜索适用于寻找最优路径,尤其是在路径长度未知或权重不同的图中。
  • 效率
    • 广度优先搜索的效率取决于图的结构,对于无权图,广度优先搜索通常能找到最短路径。
    • 启发式搜索在有启发信息的情况下可以更高效地找到最优路径,但它可能需要更复杂的评估函数。
打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP