אלגוריתם חיפוש A*

מתוך ויקיפדיה, האנציקלופדיה החופשית

אלגוריתם חיפוש *Aאנגלית: A* Search Algorithm, מבוטא "A-Star", "איי-סטאר") הוא אלגוריתם חיפוש מונחה היוריסטיקה על צומתי גרף ממושקל, תוך חיפוש צומת המקיים תכונה מסוימת (צומת היעד) במרחק הקצר ביותר מן המקור. האלגוריתם תואר לראשונה ב-1968 על ידי פיטר הרט, נילס נילסון וברטהם רפאל ממרכז המחקר של אוניברסיטת סטנפורד.

אלגוריתם *A נפוץ ביותר ונעשה בו שימוש נרחב לבעיות הדורשות מציאת מסלולים בגרפים.

כמו אלגוריתם חיפוש לרוחב, *A הוא שלם, כלומר הרצת אלגוריתם החיפוש מבטיחה מציאת מסלול בין צומת המקור לצומת היעד אם קיים מסלול כזה. הפונקציה ההיוריסטית צריכה להיות קבילה.

תיאור[עריכת קוד מקור | עריכה]

האלגוריתם הוא מסוג אלגוריתם חיפוש לרוחב ומוצא את המסלול הזול ביותר מצומת המקור לאחד מצומתי היעד. כאשר *A עובר על הגרף הוא מנסה לפתח בכל פעם את המסלול שנראה הזול ביותר תוך כדי שימוש בתור עדיפויות ממוין. האלגוריתם משתמש בפונקציית עלות המשלבת היוריסטיקה () בצירוף מידע ידוע () כדי לבחור איזה צומת לפתח. פונקציית העלות היא הסכום של שתי הפונקציות:

  • שהיא המרחק הידוע בין צומת ההתחלה לצומת הנוכחי x.
  • שהיא הערכה למרחק בין x לבין צומת היעד.

כדי שהאלגוריתם יחזיר פתרון אופטימלי, פונקציית ההיוריסטיקה, , צריכה להיות אדמיסיבלית (קבילה), כלומר לא יותר יקרה מהמרחק האמיתי לצומת היעד. בניתוב היא יכולה להיות מרחק אווירי, מכיוון שהוא מהווה את המרחק המינימלי בין כל שני צמתים. בנוסף אם ההיוריסטיקה מונוטונית (קונסיסטנטית), כלומר מקיימת את התנאי לכל קשת (x,y) בגרף (d מסמן את אורך הקשת), האלגוריתם ירוץ מהר יותר מכיוון שאין צורך לבדוק בשנית צמתים בהם האלגוריתם כבר עבר.

אפשר להסתכל על האלגוריתם כהרחבה של אלגוריתם דייקסטרה עם היוריסטיקה (בדייקסטרה הפונקציה ההיוריסטית תהיה תמיד שווה ל-0).

פסאודו-קוד[עריכת קוד מקור | עריכה]

להלן פסאודו-קוד של האלגוריתם:

function A*(start,goal)
 closedset := the empty set // The set of nodes already evaluated.
 openset := {start} // The set of tentative nodes to be evaluated, initially containing the start node
 came_from := the empty map // The map of navigated nodes.

 g_score[start] := 0 // Cost from start along best known path.
 // Estimated total cost from start to goal through y.
 f_score[start] := g_score[start] + heuristic_cost_estimate(start, goal)
 
 while openset is not empty
   current := the node in openset having the lowest f_score[] value
   if current = goal
     return reconstruct_path(came_from, goal)
 
   remove current from openset
   add current to closedset
   for each neighbor in neighbor_nodes(current)
     tentative_g_score := g_score[current] + dist_between(current,neighbor)
     if neighbor in closedset or tentative_g_score >= g_score[neighbor]     // See remark below
       continue
     if neighbor not in openset or tentative_g_score < g_score[neighbor] 
       came_from[neighbor] := current
       g_score[neighbor] := tentative_g_score
       f_score[neighbor] := g_score[neighbor] + heuristic_cost_estimate(neighbor, goal)
       if neighbor not in openset
         add neighbor to openset

 return failure

 function reconstruct_path(came_from, current_node)
   if current_node in came_from
     p := reconstruct_path(came_from, came_from[current_node])
     return (p + current_node)
   else
     return current_node

הערה: מימוש האלגוריתם מניח שפונקציית ההיוריסטיקה היא מונוטונית, ולכן מדלג על צמתים שנמצאים ב-closedset. אם ההיוריסטיקה אינה מונוטונית, יש למחוק את בדיקה זו, מאחר שערך ה-g שניתן לצומת שפותח בשלב קודם יכול להיות גדול יותר מערך ה-g האמיתי של הצומת, ולכן יש לפתח אותו מחדש (להחזיר את הצומת ל-openset).

סיבוכיות[עריכת קוד מקור | עריכה]

סיבוכיות הזמן של *A תלויה בהיורסטיקה, במקרה הגרוע ביותר, מספר הצמתים שיפותחו יהיה מעריכי ביחס לאורך הפתרון (המסלול הקצר ביותר). הסיבוכיות תהיה פולינומית אם מתקיימים התנאים הבאים: מרחב המצבים הוא עץ, קיים מצב מטרה יחיד והפונקציה ההיוריסטית h מקיימת:

כאשר h* היא ההירוסטיקה האופטימלית, כלומר העלות המדויקת בין x למטרה.