This code implements the Breadth-First Search (BFS) algorithm for searching...

April 4, 2025 at 01:40 AM

def bfs(graph, start_node, end_node): solution = [] costs = 0 frontier = [] visited = [] frontier.append(start_node) visited.append(start_node) while frontier: selected_node = frontier.pop(0) if selected_node == end_node: solution.append(selected_node) break solution.append(selected_node) for neighbour in graph[selected_node]: if neighbour not in visited: frontier.append(neighbour) visited.append(neighbour) costs += 1 return solution, costs print("Enter the graph (node -> neighbors). Type 'done' when finished:") state_space = {} while True: line = input() if line.lower() == "done": break parts = line.split() node = parts[0] neighbors = parts[1:] if len(parts) > 1 else [] state_space[node] = neighbors start_state = input("Enter the start node: ") goal_state = input("Enter the end node: ") solution, costs = bfs(state_space, start_state, goal_state) print("Solution: {}".format(solution)) print("Costs: {}".format(costs))

This code implements the Breadth-First Search (BFS) algorithm for searching a graph and returns the path explored (sequence of nodes visited) and the cost (number of iterations or steps taken during the search). Here's a step-by-step explanation of what the code does:

1. Input a graph representation:

  • The user is prompted to input the graph as a dictionary where each node is represented as a key, and the neighbors of the node are listed as values.
  • The input ends when the user types 'done'.

Example input for the graph:

A B C
B D E
C F G
D
E F
F
G H
done

This translates to:

state_space = {
    "A": ["B", "C"],
    "B": ["D", "E"],
    "C": ["F", "G"],
    "D": [],
    "E": ["F"],
    "F": [],
    "G": ["H"]
}

2. Start and Goal Nodes:

  • The user is prompted to input the start node (start_state) and the end node (goal_state) for the search.

3. Define and Run BFS:

  • The bfs() function performs the following steps:

    • Initializes:
      • solution: A list holding the sequence of explored nodes (the path).
      • costs: A counter to track the number of iterations or steps.
      • frontier: A queue to maintain the nodes to be explored (FIFO behavior).
      • visited: A list to track the nodes that have already been explored.
    • Adds the start_node to both frontier (to be explored) and visited (to avoid revisiting it).
  • Iterates while there are nodes in the frontier:

    • Pops the first node from the frontier (FIFO behavior).
    • Checks if it matches the end_node. If yes:
      • Adds it to the solution and breaks the loop (ends the search).
    • Otherwise:
      • Adds the node to the solution list and processes its neighbors from the graph.
      • For each neighbor:
        • If the neighbor has not been visited:
          • Adds it to the frontier (to explore later).
          • Marks it as visited.
    • Increments the costs counter to reflect a step in the search process.
  • Returns:

    • solution: The list of explored nodes.
    • costs: The number of iterations or steps taken in the search.

4. Execution and Output:

  • The function outputs the solution and the cost after the BFS completes its search.

Example Walkthrough:

Graph Input:

A B C
B D E
C F G
D
E F
F
G H
done

Start Node:

A

End Node:

F

BFS Process:

  1. Initialize:

    • frontier = ["A"]
    • visited = ["A"]
    • solution = []
  2. First loop iteration:

    • frontier = []
    • selected_node = "A"
    • Add "A" to solution.
    • Explore neighbors of "A" ("B", "C") and add them to frontier and visited.
    • solution = ["A"]
    • frontier = ["B", "C"]
    • visited = ["A", "B", "C"]
  3. Second loop iteration:

    • frontier = ["C"]
    • selected_node = "B"
    • Add "B" to solution.
    • Explore neighbors of "B" ("D", "E") and add them to frontier and visited.
    • solution = ["A", "B"]
    • frontier = ["C", "D", "E"]
    • visited = ["A", "B", "C", "D", "E"]
  4. Third loop iteration:

    • frontier = ["D", "E"]
    • selected_node = "C"
    • Add "C" to solution.
    • Explore neighbors of "C" ("F", "G") and add them to frontier and visited.
    • solution = ["A", "B", "C"]
    • frontier = ["D", "E", "F", "G"]
    • visited = ["A", "B", "C", "D", "E", "F", "G"]
  5. Fourth loop iteration:

    • frontier = ["E", "F", "G"]
    • selected_node = "D"
    • Add "D" to solution.
    • No new neighbors to explore (already visited).
    • solution = ["A", "B", "C", "D"]
  6. Fifth loop iteration:

    • frontier = ["F", "G"]
    • selected_node = "E"
    • Add "E" to solution.
    • No new neighbors to explore (already visited).
    • solution = ["A", "B", "C", "D", "E"]
  7. Sixth loop iteration:

    • frontier = ["G"]
    • selected_node = "F"
    • Add "F" to solution.
    • Match "F" == "goal_node". Break the loop.

Output:

Solution: ["A", "B", "C", "D", "E", "F"]
Costs: 5

Key Points:

  • This constructs the path traversed by the BFS algorithm, not necessarily the shortest path.
  • The cost reflects the number of iterations, not edge weights or distance.
Generate your own explanations
Download our vscode extension
Read other generated explanations

Built by @thebuilderjr
Sponsored by beam analytics
Read our terms and privacy policy
Forked from openai-quickstart-node