52700.fb2 Учебник по Haskell - читать онлайн бесплатно полную версию книги . Страница 308

Учебник по Haskell - читать онлайн бесплатно полную версию книги . Страница 308

В функции search нам необходимо обойти все элементы в порядке значения эвристики и остановиться

в вершине, на которой целевой предикат вернёт True. Но для начала мы добавим к вершинам их пути из

корня, для того чтобы в конце мы смогли узнать как мы попали в текущую вершину. Итак наша функция

разбивается на три составляющие:

search :: (Ord h, Ord a) => (a -> Bool) -> Tree (a, h) -> Maybe [a]

search isGoal =

findPath isGoal . flattenTree . addPath

выпишем типы составляющих функций и проверим код в интерпретаторе.

un = undefined

findPath :: (a -> Bool) -> [Path a] -> Maybe [a]

findPath = un

flattenTree :: (Ord h, Ord a) => Tree (Path a, h) -> [Path a]

flattenTree = un

addPath :: Tree (a, h) -> Tree (Path a, h)

addPath = un

data Path a = Path

{ pathEnd

:: a

, path

:: [a]

}

Обратите внимание на то как поступающие на вход данные разделились между функциями. Информа-

ция о приоритете вершин не идёт дальше функции flattenTree, а предикат isGoal используется только в

функции findPath. Модуль прошёл проверку типов и мы можем детализировать функции дальше:

addPath :: Tree (a, h) -> Tree (Path a, h)

addPath = iter []

where iter ps t = Node (Path val (reverse ps’), h) $

iter ps’ <$> subForest t

where (val, h)

= rootLabel t

ps’

= val : ps

В этой функции мы просто присоединяем к данной вершине все родительские вершины, так мы составля-

ем маршрут от данной вершины до начальной, поскольку мы всё время добавляем новые вершины в начало

списка, в итоге у нас получаются перевёрнутые маршруты, поэтому перед тем как обернуть значение в кон-

структор Path мы переворачиваем список. На самом деле нам нужно перевернуть только один путь. Путь,

который ведёт к цели, но за счёт того, что язык у нас ленивый, функция reverse будет применена не сразу, а

лишь тогда, когда нам действительно понадобится значение пути. Это как раз и произойдёт лишь один раз,

в самом конце программы, лишь для одного значения!

Давайте пока пропустим функцию flattenTree и сначала определим функцию findPath. Эта функция

принимает все вершины, которые мы обошли если бы шли без цели (функции isGoal) и ищет среди них

первую, которая удовлетворяет предикату. Для этого мы воспользуемся стандартной функцией find из мо-

дуля Data.List:

findPath :: (a -> Bool) -> [Path a] -> Maybe [a]

findPath isGoal =

fmap path . find (isGoal . pathEnd)

Напомню тип функции find, она принимает предикат и список, а возвращает первое значение списка, на

котором предикат вернёт True:

find :: (a -> Bool) -> [a] -> Maybe a

278 | Глава 19: Ориентируемся по карте