A*: различия между версиями
[отпатрулированная версия] | [отпатрулированная версия] |
Ссылка на сайте на статью поменялась. Изменил на новую. |
Lockal (обсуждение | вклад) |
||
Строка 1: | Строка 1: | ||
{{Алгоритмы поиска на графах}} |
{{Алгоритмы поиска на графах}} |
||
'''Поиск A*''' (произносится «А |
'''Поиск A*''' (произносится «А звёздочка») — в [[Информатика|информатике]] и [[Математика|математике]], [[Алгоритм|алгоритм]] поиска [[Поиск по первому наилучшему совпадению|по первому наилучшему совпадению]] на [[Граф (математика)|графе]], который находит маршрут с наименьшей стоимостью от одной вершины (начальной) к другой (целевой, конечной). |
||
Порядок обхода вершин определяется '''[[Эвристика|эвристической функцией]]''' «расстояние + стоимость» (обычно обозначаемой как ''f(x)''). Эта функция |
Порядок обхода вершин определяется '''[[Эвристика|эвристической функцией]]''' «расстояние + стоимость» (обычно обозначаемой как ''f(x)''). Эта функция — сумма двух других: функции стоимости достижения рассматриваемой вершины (''x'') из начальной (обычно обозначается как ''g(x)'' и может быть как эвристической, так и нет) и эвристической оценкой расстояния от рассматриваемой вершины к конечной (обозначается как ''h(x)''). |
||
Функция ''h(x)'' должна быть '''допустимой эвристической оценкой''', то есть не должна переоценивать расстояния к целевой вершине. Например, для задачи маршрутизации ''h(x)'' может представлять собой расстояние до цели по прямой линии, так как это физически наименьшее возможное расстояние между двумя точками. |
Функция ''h(x)'' должна быть '''допустимой эвристической оценкой''', то есть не должна переоценивать расстояния к целевой вершине. Например, для задачи маршрутизации ''h(x)'' может представлять собой расстояние до цели по прямой линии, так как это физически наименьшее возможное расстояние между двумя точками. |
||
Строка 8: | Строка 8: | ||
Этот алгоритм был впервые описан в [[1968 год]]у [[Харт, Питер|Питером Хартом]], [[Нильсон, Нильс|Нильсом Нильсоном]] и [[Рафаэль, Бертрам|Бертрамом Рафаэлем]]. В их работе он упоминается как «алгоритм A». Но так как он вычисляет лучший маршрут для заданной эвристики, он был назван A*. |
Этот алгоритм был впервые описан в [[1968 год]]у [[Харт, Питер|Питером Хартом]], [[Нильсон, Нильс|Нильсом Нильсоном]] и [[Рафаэль, Бертрам|Бертрамом Рафаэлем]]. В их работе он упоминается как «алгоритм A». Но так как он вычисляет лучший маршрут для заданной эвристики, он был назван A*. |
||
Обобщением для него является [[Двунаправленный поиск|двунаправленный эвристический алгоритм |
Обобщением для него является [[Двунаправленный поиск|двунаправленный эвристический алгоритм поиска]]. |
||
== Описание алгоритма == |
== Описание алгоритма == |
||
A* пошагово просматривает все пути, ведущие от начальной вершины в конечную, пока не найдёт минимальный. Как и все '''[[Информированный поиск|информированные алгоритмы поиска]]''', он просматривает сначала те маршруты, которые «кажутся» ведущими к цели. От [[Жадный алгоритм|жадного алгоритма]] (который тоже является алгоритмом поиска по первому лучшему совпадению) его отличает то, что при выборе вершины он учитывает, помимо прочего, ''весь'' пройденный до неё путь (составляющая ''g(x)'' |
A* пошагово просматривает все пути, ведущие от начальной вершины в конечную, пока не найдёт минимальный. Как и все '''[[Информированный поиск|информированные алгоритмы поиска]]''', он просматривает сначала те маршруты, которые «кажутся» ведущими к цели. От [[Жадный алгоритм|жадного алгоритма]] (который тоже является алгоритмом поиска по первому лучшему совпадению) его отличает то, что при выборе вершины он учитывает, помимо прочего, ''весь'' пройденный до неё путь (составляющая ''g(x)'' — это стоимость пути от начальной вершины, а не от предыдущей, как в жадном алгоритме). |
||
В начале работы просматриваются узлы, смежные с начальной; выбирается тот из них, который имеет минимальное значение ''f(x)'', после чего этот узел раскрывается. На каждом этапе алгоритм оперирует с множеством путей из начальной точки до всех ещё не раскрытых (листовых) вершин графа («множеством частных решений»), которое размещается в [[Очередь с приоритетом|очереди с приоритетом]]. Приоритет пути определяется по значению ''f(x) = g(x) + h(x)''. Алгоритм продолжает свою работу до тех пор, пока значение ''f(x)'' целевой вершины не окажется меньшим, чем любое значение в очереди (либо пока всё дерево не будет просмотрено). Из множественных решений выбирается решение с наименьшей стоимостью. |
В начале работы просматриваются узлы, смежные с начальной; выбирается тот из них, который имеет минимальное значение ''f(x)'', после чего этот узел раскрывается. На каждом этапе алгоритм оперирует с множеством путей из начальной точки до всех ещё не раскрытых (листовых) вершин графа («множеством частных решений»), которое размещается в [[Очередь с приоритетом|очереди с приоритетом]]. Приоритет пути определяется по значению ''f(x) = g(x) + h(x)''. Алгоритм продолжает свою работу до тех пор, пока значение ''f(x)'' целевой вершины не окажется меньшим, чем любое значение в очереди (либо пока всё дерево не будет просмотрено). Из множественных решений выбирается решение с наименьшей стоимостью. |
||
Строка 42: | Строка 42: | ||
Как и алгоритм [[Поиск в ширину|поиска в ширину]], A* является '''полным''' в том смысле, что он всегда находит решение, если таковое существует. |
Как и алгоритм [[Поиск в ширину|поиска в ширину]], A* является '''полным''' в том смысле, что он всегда находит решение, если таковое существует. |
||
Если эвристическая функция ''h'' допустима, то есть никогда ''не переоценивает'' действительную минимальную стоимость достижения цели, то A* сам является допустимым (или '''оптимальным'''), также при условии, что мы не отсекаем пройденные вершины. Если же мы это делаем, то для оптимальности алгоритма требуется, чтобы ''h(x)'' была ещё и '''монотонной''', или '''преемственной''' эвристикой. Свойство монотонности означает, что если существуют пути '''A—B—C''' и '''A—C''' (не обязательно через '''B'''), то оценка стоимости пути от '''A''' до '''C''' должна быть меньше либо равна сумме оценок путей '''A—B''' и '''B—C'''. (Монотонность также известна как [[неравенство треугольника]]: одна сторона [[треугольник]]а не может быть длиннее, чем сумма двух других сторон.) Математически, для всех путей ''x'', ''y'' (где ''y'' |
Если эвристическая функция ''h'' допустима, то есть никогда ''не переоценивает'' действительную минимальную стоимость достижения цели, то A* сам является допустимым (или '''оптимальным'''), также при условии, что мы не отсекаем пройденные вершины. Если же мы это делаем, то для оптимальности алгоритма требуется, чтобы ''h(x)'' была ещё и '''монотонной''', или '''преемственной''' эвристикой. Свойство монотонности означает, что если существуют пути '''A—B—C''' и '''A—C''' (не обязательно через '''B'''), то оценка стоимости пути от '''A''' до '''C''' должна быть меньше либо равна сумме оценок путей '''A—B''' и '''B—C'''. (Монотонность также известна как [[неравенство треугольника]]: одна сторона [[треугольник]]а не может быть длиннее, чем сумма двух других сторон.) Математически, для всех путей ''x'', ''y'' (где ''y'' — потомок ''x'') выполняется: |
||
: <math>g(x) + h(x) \le g(y) + h(y).</math> |
: <math>g(x) + h(x) \le g(y) + h(y).</math> |
||
Строка 51: | Строка 51: | ||
=== Особые случаи === |
=== Особые случаи === |
||
В общем говоря, [[поиск в глубину]] и [[поиск в ширину]] являются двумя частными случаями алгоритма A*. Для поиска в глубину возьмём глобальную [[Переменная (программирование)|переменную]]-счётчик '''С''', инициализировав её неким большим значением. Всякий раз при раскрытии вершины будем присваивать значение счётчика смежным вершинам, уменьшая его на единицу после каждого присваивания. Таким образом, чем раньше будет открыта вершина, тем большее значение ''h(x)'' она получит, а значит, будет просмотрена в последнюю очередь. Если положить ''h(x) = 0'' для всех вершин, то мы получим ещё один специальный случай |
В общем говоря, [[поиск в глубину]] и [[поиск в ширину]] являются двумя частными случаями алгоритма A*. Для поиска в глубину возьмём глобальную [[Переменная (программирование)|переменную]]-счётчик '''С''', инициализировав её неким большим значением. Всякий раз при раскрытии вершины будем присваивать значение счётчика смежным вершинам, уменьшая его на единицу после каждого присваивания. Таким образом, чем раньше будет открыта вершина, тем большее значение ''h(x)'' она получит, а значит, будет просмотрена в последнюю очередь. Если положить ''h(x) = 0'' для всех вершин, то мы получим ещё один специальный случай — [[алгоритм Дейкстры]]. |
||
=== Тонкости реализации === |
=== Тонкости реализации === |
||
Существует несколько особенностей [[Реализация алгоритма|реализации]] и приёмов, которые могут значительно повлиять на эффективность алгоритма. Первое, на что не мешает обратить внимание |
Существует несколько особенностей [[Реализация алгоритма|реализации]] и приёмов, которые могут значительно повлиять на эффективность алгоритма. Первое, на что не мешает обратить внимание — это то, как очередь с приоритетом обрабатывает связи между вершинами. Если вершины добавляются в неё так, что очередь работает по принципу [[LIFO]], то в случае вершин с одинаковой оценкой A* «пойдёт» в глубину. Если же при добавлении вершин реализуется принцип [[FIFO]], то для вершин с одинаковой оценкой алгоритм, напротив, будет реализовывать поиск в ширину. В некоторых случаях это обстоятельство может оказывать существенное значение на [[производительность]]. |
||
В случае, если по окончании работы от алгоритма требуется маршрут, вместе с каждой вершиной обычно хранят [[Ссылка (программирование)|ссылку]] на родительский узел. Эти ссылки позволяют реконструировать оптимальный маршрут. Если так, тогда важно, чтобы одна и та же вершина не встречалась в очереди дважды (имея при этом свой маршрут и свою оценку стоимости). Обычно для решения этой проблемы при добавлении вершины проверяют, нет ли записи о ней в очереди. Если она есть, то запись обновляют так, чтобы она соответствовала минимальной стоимости из двух. Для поиска вершины в сортирующем дереве многие стандартные алгоритмы требуют времени O(n). Если усовершенствовать дерево с помощью [[Хеш-таблица|хеш-таблицы]], то можно уменьшить это время. |
В случае, если по окончании работы от алгоритма требуется маршрут, вместе с каждой вершиной обычно хранят [[Ссылка (программирование)|ссылку]] на родительский узел. Эти ссылки позволяют реконструировать оптимальный маршрут. Если так, тогда важно, чтобы одна и та же вершина не встречалась в очереди дважды (имея при этом свой маршрут и свою оценку стоимости). Обычно для решения этой проблемы при добавлении вершины проверяют, нет ли записи о ней в очереди. Если она есть, то запись обновляют так, чтобы она соответствовала минимальной стоимости из двух. Для поиска вершины в сортирующем дереве многие стандартные алгоритмы требуют времени O(n). Если усовершенствовать дерево с помощью [[Хеш-таблица|хеш-таблицы]], то можно уменьшить это время. |
||
Строка 71: | Строка 71: | ||
: <math>|h(x) - h^*(x)| \leq O(\log h^*(x))</math>; |
: <math>|h(x) - h^*(x)| \leq O(\log h^*(x))</math>; |
||
где h* |
где h* — оптимальная эвристика, то есть точная оценка расстояния из вершины x к цели. Другими словами, ошибка ''h(x)'' не должна расти быстрее, чем [[логарифм]] от оптимальной эвристики. |
||
Но ещё бо́льшую проблему, чем временна́я сложность, представляют собой потребляемые алгоритмом ресурсы [[Оперативная память|памяти]]. В худшем случае ему приходится помнить [[Экспонента|экспоненциальное]] количество узлов. Для борьбы с этим было предложено несколько вариаций алгоритма, таких как [[алгоритм A* с итеративным углублением]] (iterative deeping A*, IDA*), [[Алгоритм A* с ограничением памяти|A* с ограничением памяти]] (memory-bounded A*, MA*), [[упрощённый MA*]] (simplified MA*, SMA*) и [[рекурсивный поиск по первому наилучшему совпадению]] (recursive best-first search, RBFS). |
Но ещё бо́льшую проблему, чем временна́я сложность, представляют собой потребляемые алгоритмом ресурсы [[Оперативная память|памяти]]. В худшем случае ему приходится помнить [[Экспонента|экспоненциальное]] количество узлов. Для борьбы с этим было предложено несколько вариаций алгоритма, таких как [[алгоритм A* с итеративным углублением]] (iterative deeping A*, IDA*), [[Алгоритм A* с ограничением памяти|A* с ограничением памяти]] (memory-bounded A*, MA*), [[упрощённый MA*]] (simplified MA*, SMA*) и [[рекурсивный поиск по первому наилучшему совпадению]] (recursive best-first search, RBFS). |
||
Строка 152: | Строка 152: | ||
}} |
}} |
||
== Ссылки == |
|||
== Внешние ссылки == |
|||
* [http://www.policyalmanac.org/games/aStarTutorial_rus.htm A* Подробное описание |
* [http://www.policyalmanac.org/games/aStarTutorial_rus.htm A* Подробное описание алгоритма А*] |
||
* [http://www.gamasutra.com/view/feature/3096/toward_more_realistic_pathfinding.php A* построение пути со сглаживанием и реалистичными поворотами] |
* [http://www.gamasutra.com/view/feature/3096/toward_more_realistic_pathfinding.php A* построение пути со сглаживанием и реалистичными поворотами] — Автор Marco Pinter |
||
* [http://www.generation5.org/content/2002/ase.asp A* Explorer] |
* [http://www.generation5.org/content/2002/ase.asp A* Explorer] — программа для Windows, позволяющая пошагово просматривать работу A* алгоритма. |
||
* [http://www.tarikattar.com/napier/osmastermap/ Алгоритм A*] |
* [http://www.tarikattar.com/napier/osmastermap/ Алгоритм A*] — реализация на PHP и визуализация используя Google Map API. |
||
* [http://theory.stanford.edu/~amitp/GameProgramming/ |
* [http://theory.stanford.edu/~amitp/GameProgramming/ Amit’s Pages] — О поиске пути и A* |
||
* [http://herbert.gandraxa.com/herbert/pfa.asp Реализация на VB 6.0] в виде DLL, включает графический интерфейс и возможность пошагового просмотра. |
* [http://herbert.gandraxa.com/herbert/pfa.asp Реализация на VB 6.0] в виде DLL, включает графический интерфейс и возможность пошагового просмотра. |
||
* [http://www.geocities.com/jheyesjones/astar.html Руководство по A*] |
* [http://www.geocities.com/jheyesjones/astar.html Руководство по A*] |
Версия от 08:45, 11 апреля 2010
Поиск A* (произносится «А звёздочка») — в информатике и математике, алгоритм поиска по первому наилучшему совпадению на графе, который находит маршрут с наименьшей стоимостью от одной вершины (начальной) к другой (целевой, конечной).
Порядок обхода вершин определяется эвристической функцией «расстояние + стоимость» (обычно обозначаемой как f(x)). Эта функция — сумма двух других: функции стоимости достижения рассматриваемой вершины (x) из начальной (обычно обозначается как g(x) и может быть как эвристической, так и нет) и эвристической оценкой расстояния от рассматриваемой вершины к конечной (обозначается как h(x)).
Функция h(x) должна быть допустимой эвристической оценкой, то есть не должна переоценивать расстояния к целевой вершине. Например, для задачи маршрутизации h(x) может представлять собой расстояние до цели по прямой линии, так как это физически наименьшее возможное расстояние между двумя точками.
Этот алгоритм был впервые описан в 1968 году Питером Хартом, Нильсом Нильсоном и Бертрамом Рафаэлем. В их работе он упоминается как «алгоритм A». Но так как он вычисляет лучший маршрут для заданной эвристики, он был назван A*.
Обобщением для него является двунаправленный эвристический алгоритм поиска.
Описание алгоритма
A* пошагово просматривает все пути, ведущие от начальной вершины в конечную, пока не найдёт минимальный. Как и все информированные алгоритмы поиска, он просматривает сначала те маршруты, которые «кажутся» ведущими к цели. От жадного алгоритма (который тоже является алгоритмом поиска по первому лучшему совпадению) его отличает то, что при выборе вершины он учитывает, помимо прочего, весь пройденный до неё путь (составляющая g(x) — это стоимость пути от начальной вершины, а не от предыдущей, как в жадном алгоритме).
В начале работы просматриваются узлы, смежные с начальной; выбирается тот из них, который имеет минимальное значение f(x), после чего этот узел раскрывается. На каждом этапе алгоритм оперирует с множеством путей из начальной точки до всех ещё не раскрытых (листовых) вершин графа («множеством частных решений»), которое размещается в очереди с приоритетом. Приоритет пути определяется по значению f(x) = g(x) + h(x). Алгоритм продолжает свою работу до тех пор, пока значение f(x) целевой вершины не окажется меньшим, чем любое значение в очереди (либо пока всё дерево не будет просмотрено). Из множественных решений выбирается решение с наименьшей стоимостью.
Чем меньше эвристика f(x), тем больше приоритет (поэтому для реализации очереди можно использовать сортирующие деревья).
function A*(start,goal)
% множество уже пройденных вершин
var closed := the empty set
% множество частных решений
var q := make_queue(path(start))
while q is not empty
var p := remove_first(q)
var x := the last node of p
if x in closed
continue
if x = goal
return p
add x to closed
% добавляем смежные вершины
foreach y in successors(x)
enqueue(q, p, y)
return failure
Множество уже пройденных вершин можно опустить (при этом алгоритм превратится в поиск по дереву решений), если известно, что решение существует, либо если структура successors
умеет отсекать циклы.
Свойства
Как и алгоритм поиска в ширину, A* является полным в том смысле, что он всегда находит решение, если таковое существует.
Если эвристическая функция h допустима, то есть никогда не переоценивает действительную минимальную стоимость достижения цели, то A* сам является допустимым (или оптимальным), также при условии, что мы не отсекаем пройденные вершины. Если же мы это делаем, то для оптимальности алгоритма требуется, чтобы h(x) была ещё и монотонной, или преемственной эвристикой. Свойство монотонности означает, что если существуют пути A—B—C и A—C (не обязательно через B), то оценка стоимости пути от A до C должна быть меньше либо равна сумме оценок путей A—B и B—C. (Монотонность также известна как неравенство треугольника: одна сторона треугольника не может быть длиннее, чем сумма двух других сторон.) Математически, для всех путей x, y (где y — потомок x) выполняется:
A* также оптимально эффективен для заданной эвристики h. Это значит, что любой другой алгоритм исследует не меньше узлов, чем A* (за исключением случаев, когда существует несколько частных решений с одинаковой эвристикой, точно соответствующей стоимости оптимального пути).
В то время как A* оптимален для «случайно» заданных графов, нет гарантии, что он сделает свою работу лучше, чем более простые, но и более информированные относительно проблемной области алгоритмы. Например, в неком лабиринте может потребоваться сначала идти по направлению от выхода, и только потом повернуть назад. В этом случае обследование вначале тех вершин, которые расположены ближе к выходу (по прямой дистанции), будет потерей времени.
Особые случаи
В общем говоря, поиск в глубину и поиск в ширину являются двумя частными случаями алгоритма A*. Для поиска в глубину возьмём глобальную переменную-счётчик С, инициализировав её неким большим значением. Всякий раз при раскрытии вершины будем присваивать значение счётчика смежным вершинам, уменьшая его на единицу после каждого присваивания. Таким образом, чем раньше будет открыта вершина, тем большее значение h(x) она получит, а значит, будет просмотрена в последнюю очередь. Если положить h(x) = 0 для всех вершин, то мы получим ещё один специальный случай — алгоритм Дейкстры.
Тонкости реализации
Существует несколько особенностей реализации и приёмов, которые могут значительно повлиять на эффективность алгоритма. Первое, на что не мешает обратить внимание — это то, как очередь с приоритетом обрабатывает связи между вершинами. Если вершины добавляются в неё так, что очередь работает по принципу LIFO, то в случае вершин с одинаковой оценкой A* «пойдёт» в глубину. Если же при добавлении вершин реализуется принцип FIFO, то для вершин с одинаковой оценкой алгоритм, напротив, будет реализовывать поиск в ширину. В некоторых случаях это обстоятельство может оказывать существенное значение на производительность.
В случае, если по окончании работы от алгоритма требуется маршрут, вместе с каждой вершиной обычно хранят ссылку на родительский узел. Эти ссылки позволяют реконструировать оптимальный маршрут. Если так, тогда важно, чтобы одна и та же вершина не встречалась в очереди дважды (имея при этом свой маршрут и свою оценку стоимости). Обычно для решения этой проблемы при добавлении вершины проверяют, нет ли записи о ней в очереди. Если она есть, то запись обновляют так, чтобы она соответствовала минимальной стоимости из двух. Для поиска вершины в сортирующем дереве многие стандартные алгоритмы требуют времени O(n). Если усовершенствовать дерево с помощью хеш-таблицы, то можно уменьшить это время.
Почему A* допустим и оптимален
Алгоритм A* и допустим, и обходит при этом минимальное количество вершин, благодаря тому, что он работает с «оптимистичной» оценкой пути через вершину. Оптимистичной в том смысле, что, если он пойдёт через эту вершину, у алгоритма «есть шанс», что реальная стоимость результата будет равна этой оценке, но никак не меньше. Но, поскольку A* является информированным алгоритмом, такое равенство может быть вполне возможным.
Когда A* завершает поиск, он, согласно определению, нашёл путь, истинная стоимость которого меньше, чем оценка стоимости любого пути через любой открытый узел. Но поскольку эти оценки являются оптимистичными, соответствующие узлы можно без сомнений отбросить. Иначе говоря, A* никогда не упустит возможности минимизировать длину пути, и потому является допустимым.
Предположим теперь, что некий алгоритм B вернул в качестве результата путь, длина которого больше оценки стоимости пути через некоторую вершину. На основании эвристической информации, для алгоритма B нельзя исключить возможность, что этот путь имел и меньшую реальную длину, чем результат. Соответственно, пока алгоритм B просмотрел меньше вершин, чем A*, он не будет допустимым. Итак, A* проходит наименьшее количество вершин графа среди допустимых алгоритмов, использующих такую же точную (или менее точную) эвристику.
Оценка сложности
Временна́я сложность алгоритма A* зависит от эвристики. В худшем случае, число вершин, исследуемых алгоритмом, растёт экспоненциально по сравнению с длиной оптимального пути, но сложность становится полиномиальной, когда эвристика удовлетворяет следующему условию:
- ;
где h* — оптимальная эвристика, то есть точная оценка расстояния из вершины x к цели. Другими словами, ошибка h(x) не должна расти быстрее, чем логарифм от оптимальной эвристики.
Но ещё бо́льшую проблему, чем временна́я сложность, представляют собой потребляемые алгоритмом ресурсы памяти. В худшем случае ему приходится помнить экспоненциальное количество узлов. Для борьбы с этим было предложено несколько вариаций алгоритма, таких как алгоритм A* с итеративным углублением (iterative deeping A*, IDA*), A* с ограничением памяти (memory-bounded A*, MA*), упрощённый MA* (simplified MA*, SMA*) и рекурсивный поиск по первому наилучшему совпадению (recursive best-first search, RBFS).
Литература
- Лорьер Ж.-Л. Системы искусственного интеллекта / Пер. с фр. и ред. В. Л. Стефанюка. — М.: Мир, 1991. — С. 238—244. — 20 000 экз. экз. — ISBN 5-03-001408-X.
- Рассел С. Дж., Норвиг, П. Искусственный интеллект: современный подход = Artificial Intelligence: A Modern Approach / Пер. с англ. и ред. К. А. Птицына. — 2-е изд.. — М.: Вильямс, 2006. — С. 157—162. — 3000 экз. экз. — ISBN 5-8459-0887-6.
- Нильсон Н. Искусственный интеллект: методы поиска решений = Problem-solving Methods in Artificial Intelligence / Пер. с англ. В. Л. Стефанюка; под ред. С. В. Фомина. — М.: Мир, 1973. — С. 70 — 80.
- Dechter, R., Pearl, J. Generalized best-first search strategies and the optimality of A* // Journal of the ACM. — 1985. — Т. 32, № 3. — С. 505 — 536.
- Hart P. E., Nilsson, N. J., Raphael, B. A Formal Basis for the Heuristic Determination of Minimum Cost Paths // IEEE Transactions on Systems Science and Cybernetics SSC4. — 1968. — № 2. — С. 100 — 107.
- Hart P. E., Nilsson, N. J., Raphael, B. Correction to «A Formal Basis for the Heuristic Determination of Minimum Cost Paths» // SIGART Newsletter. — 1972. — Т. 37. — С. 28 — 29.
- Pearl J. Heuristics: Intelligent Search Strategies for Computer Problem Solving. — Addison-Wesley, 1984. — ISBN 0-201-05594-5.
Ссылки
- A* Подробное описание алгоритма А*
- A* построение пути со сглаживанием и реалистичными поворотами — Автор Marco Pinter
- A* Explorer — программа для Windows, позволяющая пошагово просматривать работу A* алгоритма.
- Алгоритм A* — реализация на PHP и визуализация используя Google Map API.
- Amit’s Pages — О поиске пути и A*
- Реализация на VB 6.0 в виде DLL, включает графический интерфейс и возможность пошагового просмотра.
- Руководство по A*
- Cupper. Принцип работы алгоритма поиска пути Астар (A*) . GameDev.ru (17 января 2004). Дата обращения: 22 июля 2009.