Thứ Năm, 25 tháng 9, 2014

Thuật toán Floyd tìm đường đi ngắn nhất

Đường đi ngắn nhất giữa tất cả các cặp đỉnh


Rõ ràng ta có thể giải bài toán tìm đường đi ngắn nhất giữa tất cả các cặp đỉnh của đồ thị bằng cách sử dụng n lần thuật toán mô tả ở mục trước, trong đó ta sẽ chọn s lần lượt là các đỉnh của đồ thị. Rõ ràng, khi đó ta thu được thuật toán với độ phức tạp O(n4) (nếu sử dụng thuật toán Ford_Bellman) hoặc O(n3) đối với trường hợp trọng số không âm hoặc đồ thị không có chu trình. Trong trường hợp tổng quát, sử dụng thuật toán Ford_Bellman n lần không phải là cách làm tốt nhất. Ở đây ta sẽ mô tả một thuật toán giải bài toán trên với độ phức tạp tính toán O(n3): thuật toán Floyd. Thuật toán được mô tả trong thủ tục sau đây.
Procedure Floyd;
(* Tìm đường đi ngắn nhất giữa tất cả các cặp đỉnh
Đầu vào: Đồ thị cho bởi ma trận trọng số a[i,j], i, j =1, 2,. . ,n.
Đầu ra:
Ma trận đường đi ngắn nhất giữa các cặp đỉnh
d[i,j]=, i,j = 1, 2. . .,n,
trong đó d[i,j] cho độ dài đường đi ngắn nhất từ đỉnh i đến đỉnh j.
Ma trận ghi nhận đường đi
p[i,j], i, j = 1, 2.. . , n,
trong đó p[i,j] ghi nhận đỉnh đi trước đỉnh j trong đường đi ngắn nhất từ i đến j.
*)
begin
                  (* Khởi tạo *)
                  for i:=1 to n do
                       for j:=1 to n do
                       begin
                         d[i,j]:=a[i.j];
                          p[i.j]:=i;
                      end;
                 (* Bước lặp *)
                 for k:=1 to n do
                     for i:=1 to n do
                             for j:=1 to n do
                                if d[i,j]>d[i,k]+d[k,j] then
                               begin
                                      d[i,j]+d[i,k]+d[k,j];
                                      p[i,j]>p[k,j];
                               end;
end;
Rõ ràng độ phức tạp tính toán của thuật toán là O(n3).
Kết thúc chương này chúng ra trình bày một cách thể hiện thuật toán Dijkstra trên ngôn ngữ Pascal:

(* CHƯƠNG TRÌNH TÌM ĐƯỜNG ĐI NGẮN NHẤT TỪ ĐỈNH S ĐẾN ĐỈNH T THEO THUẬT TOÁN DIJKSTRA *)
uses crt;
        const max=50;
        var
          n, s, t:integer;
          chon:char;
          Truoc:array[1..max] of byte;
          d: array[1..max] of integer;
          a: array[1..max,1..max] of integer;
          final: array[1..max] of boolean;
procedure Nhapsolieu;
          var
          f:text;
          fname:string;
          i,j:integer;
          begin
                     write(‘Vao ten file du lieu can doc:’);
                     readln(fname);
                              assign(f,fname);
                     reset(f);
                     readln(f,n);
                              for i:=1 to n do
                                     for j:=1 to n do read(f, a[i,j];
                     close(f);
          end;
procedure Insolieu;
        var
         i,j:integer;
         begin
                      writeln(‘So dinh cua do thi:’,n);
                      writeln(‘Ma tran khoang cach:’);
                      for i:=1 to n do
                      begin
                              for j:=1 to n do write(a[i,j]:3,’ ‘);
                              writeln;
                     end;
         end;
Procedure Inketqua;
         Var
          i,j:integer;
          begin
                           writeln(‘Duong di ngan nhat tu ‘,s,’ den ‘,t);
                            write(t,’ ⇐ ’);
                            while i<> s do
                            begin
                                        i:=Truoc[i];
                                        write(i,’ ⇐ ’);
                            end;
            end;
Procedure Dijkstra;
           Var
           U,v,minp:integer;
           Begin
                                Write(‘Tim duon di tu s=’);Readln(s);Write(‘ den t=’);Readln(t);
                                For v:=1 to n do
                               Begin
                                       d[v]:=a[s,v];
                                       Truoc[v]:=s;
                                        Filal[v]:=false;
                                End;
                               Truoc[s]:=0;D[s]:=0;Final[s]:=true;
                               While not final[t] do (* Buoc lap *)
                                   Begin
                                                  Tim u la dinh co nhan tam thoi nho nhat
                                                   minp:=maxint;
                                                   for v:=1 to n do
                                                              if (not final[v]) ans minp>d[v]) then
                                                               begin
                                                                           u:=v;
                                                                           minp:=d[v];
                                                              end;
                                                  final[u]:=true;
                                                  if not final[t] then
                                                             for v:=1 to n do
                                                                   if (not final[v]) and (d[u]+a[u,v]<d[v]) then
                                                                   begin
                                                                               d[v]:=d[u]+a[u,v];
                                                                               Truoc[v]:=u;
                                                                   end;
                                   End;
                end;
Procedure Menu;
       Begin
                 Clrscr;
                 Writeln(‘1. Nhap du lieu tu file’);
                 Writeln(‘2. Giai bai toan’);
                 Writeln(‘3. Ket thuc’);
                 Writeln(‘---------------------‘);
                 Write(‘Hay chon chuc nang:’):
       End;
(* Chuong trinh chinhs *)
Begin
                     Repeat
                     Menu;
                     Chon:=readkey;
                                Writeln(chon);
                                Case chon of
                                               ‘1’:Nhapsolieu;
                                               ‘2’: begin
                                                               Insolieu;
                                                               Dijkstra;
                                                               Inketqua;
                                               ‘3’:exit;
                                                     end;
                      Until false;
End.

Source code: 
Liên hệ: vuthanhnam94@gmail.com

Không có nhận xét nào:

Đăng nhận xét

Bài đăng phổ biến