Algorithms
Breadth First Search

Breadth-First Search (BFS)

Introduction

Breadth-First Search (BFS) is a fundamental graph traversal algorithm used to explore or search through a graph or tree structure. It starts at a given node and explores all its neighbors before moving to the next level of nodes. BFS follows a breadth-first strategy, which means it explores nodes in layers, gradually moving away from the source.

Explanation

The BFS algorithm works as follows:

  1. Start at a given node (often referred to as the "source" or "root" node).
  2. Create a queue and enqueue the source node.
  3. Mark the source node as visited.
  4. Dequeue a node from the queue and print it (or perform any desired operation on the node).
  5. Enqueue all the unvisited neighbors of the dequeued node.
  6. Mark the enqueued neighbors as visited.
  7. Repeat steps 4-6 until the queue is empty.

Implementation

Here's an implementation of BFS for an undirected graph using an adjacency list representation in Python:

from collections import deque
 
def bfs(graph, start):
    visited = set()
    queue = deque([start])
    visited.add(start)
 
    while queue:
        node = queue.popleft()
        print(node, end=" ")
 
        for neighbor in graph[node]:
            if neighbor not in visited:
                visited.add(neighbor)
                queue.append(neighbor)

Step-by-step explanation of the code:

  1. The bfs function takes the graph (represented as an adjacency list) and the starting node.
  2. A set visited is initialized to keep track of the visited nodes.
  3. A deque queue is created, and the starting node is enqueued.
  4. The starting node is marked as visited.
  5. The loop continues until the queue is empty.
  6. The first node in the queue is dequeued.
  7. The value of the dequeued node is printed (or you can perform any desired operation on the node).
  8. For each unvisited neighbor of the dequeued node, it is marked as visited and enqueued.
  9. The process continues until all nodes reachable from the starting node are visited.

Use Cases

BFS has various applications, including:

  • Finding the shortest path between two nodes in an unweighted graph.
  • Testing if a graph is bipartite.
  • Finding the connected components of a graph.
  • Solving puzzles and games (e.g., solving a sliding puzzle).

Time and Space Complexity

The time complexity of BFS is O(V + E), where V is the number of vertices (nodes) and E is the number of edges in the graph. This is because BFS visits each vertex once and each edge once. The space complexity is O(V) as it requires auxiliary space for the visited set and the queue to store nodes during traversal.

Variants or Extensions

Some variants or extensions of BFS include:

  • Bidirectional BFS: A technique that simultaneously performs two BFS searches, one from the source node and another from the target node, meeting at some intermediate node. It can significantly improve efficiency for finding the shortest path in certain cases.
  • Depth-Limited Search (DLS): A modified version of BFS that limits the depth of exploration to avoid infinite loops in graphs with cycles.
  • Iterative Deepening Depth-First Search (IDDFS): A combination of depth-first search and breadth-first search that performs a series of DFS with increasing depth limits, allowing exploration in a depth-first manner while also covering all nodes at a given depth.

Summary

Breadth-First Search (BFS) is a powerful graph traversal algorithm that explores a graph or tree structure by systematically examining nodes in layers. It is widely used for graph-related problems, especially when finding the shortest path or exploring all nodes at a certain distance from the source. Understanding BFS is essential for programmers dealing with graph-related data structures and applications.