Tìm đường đi ngắn nhất trên đơn đồ thị có hướng trang 73 Chuyên đề Tin học 12

Vấn đề 1 trang 73 Chuyên đề Tin học 12: Tìm đường đi ngắn nhất trên đơn đồ thị có hướng

Một dãy đỉnh a = i0, i1,..., i= b (a ≠ b) được gọi là đường đi từ đinh a tới đỉnh b nếu hai đỉnh liên tiếp trên đường đi có cạnh nối.

Trong cuộc sống chúng ta gặp rất nhiều bài toán liên quan đến tìm đường đi, dưới đây là một ví dụ cụ thể.

Có n sân bay được đánh số từ 0 đến n - 1, có m tuyến bay giữa các sân bay, tuyến thứ k sẽ bay từ sân bay i, sang sân bay j .Em hãy tìm cách di chuyển từ sân bay a tới sân bay b với số tuyến bay là ít nhất.

Lời giải:

Một dãy đỉnh a = i0, i1,..., i= b (a ≠ b) được gọi là đường đi từ đinh a tới đỉnh b nếu hai đỉnh liên tiếp trên đường đi có cạnh nối.

Trong cuộc sống chúng ta gặp rất nhiều bài toán liên quan đến tìm đường đi, dưới đây là một ví dụ cụ thể.

Có n sân bay được đánh số từ 0 đến n - 1, có m tuyến bay giữa các sân bay, tuyến thứ k sẽ bay từ sân bay i, sang sân bay j. cách di chuyển từ sân bay a tới sân bay b với số tuyến bay là ít nhất bằng cách biểu diễn đồ thị như sau:

a) Biểu diễn đồ thị: Sử dụng danh sách kề (adjacency list) để biểu diễn đồ thị có hướng. Mỗi đỉnh (sân bay) sẽ có một danh sách các đỉnh mà nó có cạnh nối tới (các sân bay có tuyến bay trực tiếp tới).

b) Thuật toán BFS:

- BFS là thuật toán tìm kiếm theo chiều rộng, rất hiệu quả trong việc tìm đường đi ngắn nhất trên đồ thị không trọng số.

- Khởi tạo một hàng đợi (queue) và đưa đỉnh bắt đầu aaa vào hàng đợi.

- Sử dụng một mảng để theo dõi các đỉnh đã được thăm và mảng này cũng lưu khoảng cách từ đỉnh aaa tới các đỉnh khác.

- Trong khi hàng đợi không rỗng, lấy đỉnh đầu hàng đợi ra và duyệt qua tất cả các đỉnh kề của nó. Nếu đỉnh kề chưa được thăm, đánh dấu nó đã được thăm và thêm nó vào hàng đợi.

c) Truy vết đường đi:

- Để truy vết đường đi từ bbb về aaa, chúng ta có thể sử dụng một mảng cha (parent array) để lưu đỉnh trước đó của mỗi đỉnh trong quá trình BFS.

- Khi tìm thấy đỉnh bbb, chúng ta có thể truy ngược lại từ bbb về aaa để xác định đường đi.

d) Chương trình Python:

python

Sao chép mã

from collections import deque

def shortest_path(n, edges, a, b):

   # Bước 1: Xây dựng đồ thị dưới dạng danh sách kề

   graph = [[] for _ in range(n)]

   for edge in edges:

        i, j = edge

        graph[i].append(j)

   # Bước 2: BFS để tìm đường đi ngắn nhất

   queue = deque([a])

   visited = [False] * n

   distance = [-1] * n  # Khoảng cách từ a tới các đỉnh

   parent = [-1] * n  # Để truy vết đường đi

   visited[a] = True

   distance[a] = 0

   while queue:

        current = queue.popleft()

        for neighbor in graph[current]:

            if not visited[neighbor]:

                visited[neighbor] = True

                distance[neighbor] = distance[current] + 1

                parent[neighbor] = current

                queue.append(neighbor)

                if neighbor == b:

                    # Nếu đã tới đích, dừng BFS

                    queue.clear()

                    break

   # Bước 3: Truy vết đường đi từ b về a

   path = []

   if visited[b]:

        current = b

        while current != -1:

            path.append(current)

            current = parent[current]

        path.reverse()

   return path if path and path[0] == a else []

# Ví dụ sử dụng:

n = 5  # Số sân bay

edges = [(0, 1), (0, 2), (1, 2), (1, 3), (2, 3), (3, 4)]  # Các tuyến bay

a = 0  # Sân bay bắt đầu

b = 4  # Sân bay đích

print(shortest_path(n, edges, a, b))

Lời giải bài tập Chuyên đề Tin 12 Bài 6: Dự án học tập: Tìm hiểu các vấn đề ứng dụng đồ thị hay, chi tiết khác:

Xem thêm lời giải bài tập Chuyên đề học tập Tin học 12 Cánh diều hay, chi tiết khác:

Xem thêm các tài liệu học tốt lớp 12 hay khác:


Giải bài tập lớp 12 sách mới các môn học