סיבוכיות זמן | O(|n|^3) |
---|---|
סיבוכיות מקום | O(|V|^2) |
ממציא | ברנארד רוי |
נקרא על שם | רוברט פלויד, סטיבן וורשאל |
הומצא | 1959 |
אלגוריתם פלויד-וורשאל הוא אלגוריתם במדעי המחשב המשמש למציאת המסלולים הקצרים (או הקלים) ביותר בין כל שני צמתים, בגרף ממושקל ומכוון. האלגוריתם מבוסס על פרדיגמת התכנון הדינמי. האלגוריתם פועל גם על גרפים שמכילים קשתות עם משקלים שליליים, אבל שלא מכילים מעגל בעל משקל שלילי (בניגוד לאלגוריתם דייקסטרה שמניח קשתות אי שליליות). אם בגרף יש מעגל עם משקל שלילי האלגוריתם יגלה זאת. סיבוכיות זמן הריצה של האלגוריתם היא .
האלגוריתם פורסם על ידי רוברט פלויד בשנת 1962 בגרסתו המוכרת לנו כיום, אך גרסאות דומות של האלגוריתם פורסמו ב-1959 על ידי ברנארד רוי, ועל ידי סטיבן וורשאל ב-1962 במטרה לחשב את הסגור הטרנזיטיבי של קבוצת קודקודים בגרף[1].
האלגוריתם מתבסס על האבחנה הבאה: אם ממספרים את קבוצת הצמתים של הגרף כך שהיא מסומנת ומסמנים בתור את משקל המסלול הקצר ביותר מהצומת אל הצומת שעובר בצומתי ביניים השייכים אך ורק לקבוצה , אז מתקיים: .
הסבר: אם קיים מסלול קצר ביותר מ- אל שעובר בצמתים שמספרם לכל היותר לא עובר בצומת שמספרו , אז . לעומת זאת, אם כל מסלול קצר ביותר כזה חייב לעבור בצומת שמספרו אז ניתן להניח שהוא עובר בו רק פעם אחת (כי אין מעגלים שליליים בגרף, ולכן אם חוזרים לאותו צומת פעמיים ניתן להשמיט מהמסלול את המעגל מהצומת לעצמו). לכן אפשר לפרק כל מסלול כזה לשני מסלולים: אחד שהולך מ- אל , ושני שהולך מ- אל . שני המסלולים הללו לא עוברים בצמתי ביניים שמספרם גדול מ-, ולכן סכום משקלם נתון על ידי . לכן כל מה שנותר הוא לבדוק איזו משתי השיטות עדיפה לכל צומת.
כעת, פעולת האלגוריתם היא זו: הוא מבצע איטרציות, כאשר באיטרציה ה- הוא מחשב את לכל זוג צמתים ושומר את המידע בטבלה. חישוב מתבצע בזמן בהינתן המידע מהאיטרציה הקודמת, ולכן כל איטרציה של האלגוריתם מתבצעת בזמן (מספרם של זוגות הצמתים האפשריים). לכן זמן הריצה הכולל של האלגוריתם יהיה .
המחשת האלגוריתם
תהי W מטריצת משקלי הקשתות: כלומר w(i,j) הוא משקל הקשת המחברת את i עם j. (משקל זה הוא אין-סוף אם אין קשת ביניהם).
בסוף, היא מטריצת אורך המסלולים הקצרים ביותר שעוברים דרך הצמתים .
פסאודו קוד:
let dist be a |V| × |V| array of minimum distances initialized to ∞ (infinity)
for each edge (u,v)
dist[u][v] ← w(u,v) // the weight of the edge (u,v)
for each vertex v
dist[v][v] ← 0
for k from 1 to |V|
for i from 1 to |V|
for j from 1 to |V|
if dist[i][j] > dist[i][k] + dist[k][j]
dist[i][j] ← dist[i][k] + dist[k][j]
end if
end-for
end-for
end-for
מעניין לציין שגם בהרצה בסדר "הלא נכון" האלגוריתם עובד כשורה לאחר מספר חזרות, כלומר אם נחליף את השורות:
for k from 1 to |V|
for i from 1 to |V|
for j from 1 to |V|
...
end-for
end-for
end-for
באחת משתי האפשרויות:
for _ from 1 to 3
for i from 1 to |V|
for j from 1 to |V|
for k from 1 to |V|
...
end-for
end-for
end-for
end-for
for _ from 1 to 2
for i from 1 to |V|
for k from 1 to |V|
for j from 1 to |V|
...
end-for
end-for
end-for
end-for
האלגוריתם ירוץ כשורה
מעגל שלילי הוא מעגל שסכום קשתותיו קטן מאפס. בין שני קודקודים שמשתתפים במעגל שלילי לא קיים מסלול קצר ביותר, כי בכל מעבר על המעגל נקבל מסלול שמשקלו קטן יותר, וניתן להמשיך כך עד אין-קץ. על כן, במקרה של מעגל שלילי האלגוריתם לא יפיק פלט משמעותי. עם זאת, במידה וקיים מעגל שלילי, ניתן להיעזר באלגוריתם על מנת לגלות זאת:
מכאן, באמצעות התבוננות במטריצת המסלולים הקצרים ביותר שהפיק האלגוריתם, ניתן להסיק שקיים מעגל שלילי, אם באחד מהתאים על האלכסון הראשי קיים מרחק קטן מאפס.
{{cite web}}
: (עזרה)