Миран Липовача - Изучай Haskell во имя добра!
- Название:Изучай Haskell во имя добра!
- Автор:
- Жанр:
- Издательство:ДМК Пресс
- Год:2012
- Город:Москва
- ISBN:978-5-94074-749-9
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Миран Липовача - Изучай Haskell во имя добра! краткое содержание
Язык Haskell имеет множество впечатляющих возможностей, но главное его свойство в том, что меняется не только способ написания кода, но и сам способ размышления о проблемах и возможных решениях. Этим Haskell действительно отличается от большинства языков программирования. С его помощью мир можно представить и описать нестандартным образом. И поскольку Haskell предлагает совершенно новые способы размышления о проблемах, изучение этого языка может изменить и стиль программирования на всех прочих.
Ещё одно необычное свойство Haskell состоит в том, что в этом языке придаётся особое значение рассуждениям о типах данных. Как следствие, вы помещаете больше внимания и меньше кода в ваши программы.
Вне зависимости от того, в каком направлении вы намерены двигаться, путешествуя в мире программирования, небольшой заход в страну Haskell себя оправдает. А если вы решите там остаться, то наверняка найдёте чем заняться и чему поучиться!
Эта книга поможет многим читателям найти свой путь к Haskell.
Отображения, монады, моноиды и другое! Всё сказано в названии: «Изучай Хаскель во имя добра!» – весёлый иллюстрированный самоучитель по этому сложному функциональному языку.
С помощью оригинальных рисунков автора, отсылке к поп-культуре, и, самое главное, благодаря полезным примерам кода, эта книга обучает основам функционального программирования так, как вы никогда не смогли бы себе представить.
Вы начнете изучение с простого материала: основы синтаксиса, рекурсия, типы и классы типов. Затем, когда вы преуспеете в основах, начнется настоящий мастер-класс от профессионала: вы изучите, как использовать аппликативные функторы, монады, застежки, и другие легендарные конструкции Хаскеля, о которых вы читали только в сказках.
Продираясь сквозь образные (и порой безумные) примеры автора, вы научитесь:
• Смеяться в лицо побочным эффектам, поскольку вы овладеете техниками чистого функционального программирования.
• Использовать волшебство «ленивости» Хаскеля для игры с бесконечными наборами данных.
• Организовывать свои программы, создавая собственные типы, классы типов и модули.
• Использовать элегантную систему ввода-вывода Хаскеля, чтобы делиться гениальностью ваших программ с окружающим миром.
Нет лучшего способа изучить этот мощный язык, чем чтение «Изучай Хаскель во имя добра!», кроме, разве что, поедания мозга его создателей. Миран Липовача (Miran Lipovača) изучает информатику в Любляне (Словения). Помимо его любви к Хаскелю, ему нравится заниматься боксом, играть на бас-гитаре и, конечно же, рисовать. У него есть увлечение танцующими скелетами и числом 71, а когда он проходит через автоматические двери, он притворяется, что на самом деле открывает их силой своей мысли.
Изучай Haskell во имя добра! - читать онлайн бесплатно ознакомительный отрывок
Интервал:
Закладка:
Но что если тип содержимого типа Maybe
не имеет экземпляра класса Monoid
? Обратите внимание: в предыдущем объявлении экземпляра единственный случай, когда мы должны полагаться на то, что содержимые являются моноидами, – это когда оба параметра функции mappend
обёрнуты в конструктор Just
. Когда мы не знаем, являются ли содержимые моноидами, мы не можем использовать функцию mappend
между ними; так что же нам делать? Ну, единственное, что мы можем сделать, – это отвергнуть второе значение и оставить первое. Для этой цели существует тип First
a
. Вот его определение:
newtype First a = First { getFirst :: Maybe a }
deriving (Eq, Ord, Read, Show)
Мы берём тип Maybe a
и оборачиваем его с помощью декларации newtype
. Экземпляр класса Monoid
в данном случае выглядит следующим образом:
instance Monoid (Firsta) where
mempty = First Nothing
First (Just x) `mappend` _ = First (Just x)
First Nothing `mappend` x = x
Значение mempty
– это просто Nothing
, обёрнутое с помощью конструктора First
. Если первый параметр функции mappend
является значением Just
, мы игнорируем второй. Если первый параметр – Nothing
, тогда мы возвращаем второй параметр в качестве результата независимо от того, является ли он Just
или Nothing
:
ghci> getFirst $ First (Just 'a') `mappend` First (Just 'b')
Just 'a'
ghci> getFirst $ First Nothing `mappend` First (Just 'b')
Just 'b'
ghci> getFirst $ First (Just 'a') `mappend` First Nothing
Just 'a'
Тип First
полезен, когда у нас есть множество значений типа Maybe
и мы хотим знать, является ли какое-либо из них значением Just
. Для этого годится функция mconcat
:
ghci> getFirst . mconcat . map First $ [Nothing, Just 9, Just 10]
Just 9
Если нам нужен моноид на значениях Maybe a
– такой, чтобы оставался второй параметр, когда оба параметра функции mappend
являются значениями Just
, то модуль Data.Monoid
предоставляет тип Last a
, который работает, как и тип First a
, но при объединении с помощью функции mappend
и использовании функции mconcat
сохраняется последнее значение, не являющееся Nothing
:
ghci> getLast . mconcat . map Last $ [Nothing, Just 9, Just 10]
Just 10
ghci> getLast $ Last (Just "один") `mappend` Last (Just "два")
Just "two"
Свёртка на моноидах
Один из интересных способов ввести моноиды в работу заключается в том, чтобы они помогали нам определять свёртки над различными структурами данных. До сих пор мы производили свёртки только над списками, но списки – не единственная структура данных, которую можно свернуть. Мы можем определять свёртки почти над любой структурой данных. Особенно хорошо поддаются свёртке деревья.
Поскольку существует так много структур данных, которые хорошо работают со свёртками, был введён класс типов Foldable
. Подобно тому как класс Functor
предназначен для сущностей, которые можно отображать, класс Foldable
предназначен для вещей, которые могут быть свёрнуты! Его можно найти в модуле Data.Foldable
; и, поскольку он экспортирует функции, имена которых конфликтуют с именами функций из модуля Prelude
, его лучше импортировать, квалифицируя (и подавать с базиликом!):
import qualified Data.Foldable as F
Чтобы сэкономить драгоценные нажатия клавиш, мы импортировали его, квалифицируя как F
.
Так какие из некоторых функций определяет этот класс типов? Среди них есть функции foldr
, foldl
, foldr1
и foldl1
. Ну и?.. Мы уже давно знакомы с ними! Что ж в этом нового? Давайте сравним типы функции foldr
из модуля Foldable
и одноимённой функции из модуля Prelude
, чтобы узнать, чем они отличаются:
ghci> :t foldr
foldr :: (a –> b –> b) –> b –> [a] –> b
ghci> :t F.foldr
F.foldr :: (F.Foldable t) => (a –> b –> b) –> b –> t a –> b
А-а-а! Значит, в то время как функция foldr
принимает список и сворачивает его, функция foldr
из модуля Data.Foldable
принимает любой тип, который можно свернуть, – не только списки! Как и ожидалось, обе функции foldr
делают со списками одно и то же:
ghci> foldr (*) 1 [1,2,3]
6
ghci> F.foldr (*) 1 [1,2,3]
6
Другой структурой данных, поддерживающей свёртку, является Maybe
, которую мы все знаем и любим!
ghci> F.foldl (+) 2 (Just 9)
11
ghci> F.foldr (||) False (Just True)
True
Но сворачивание значения Maybe
не очень-то интересно. Оно действует просто как список с одним элементом, если это значение Just
, и как пустой список, если это значение Nothing
. Давайте рассмотрим чуть более сложную структуру данных.
Помните древовидную структуру данных из главы 7? Мы определили её так:
data Tree a = EmptyTree | Node a (Tree a) (Tree a) deriving (Show)
Вы узнали, что дерево – это либо пустое дерево, которое не содержит никаких значений, либо узел, который содержит одно значение, а также два других дерева. После того как мы его определили, мы сделали для него экземпляр класса Functor
, и это дало нам возможность отображать его с помощью функций, используя функцию fmap
. Теперь мы определим для него экземпляр класса Foldable
, чтобы у нас появилась возможность производить его свёртку.
Один из способов сделать для конструктора типа экземпляр класса Foldable
состоит в том, чтобы просто напрямую реализовать для него функцию foldr
. Но другой, часто более простой способ состоит в том, чтобы реализовать функцию foldMap
, которая также является методом класса типов Foldable
. У неё следующий тип:
foldMap :: (Monoid m, Foldable t) => (a –> m) –> t a –> m
Её первым параметром является функция, принимающая значение того типа, который содержит наша сворачиваемая структура (обозначен здесь как a
), и возвращающая моноидное значение. Второй её параметр – сворачиваемая структура, содержащая значения типа a
. Эта функция отображает структуру с помощью заданной функции, таким образом, производя сворачиваемую структуру, которая содержит моноидные значения. Затем, объединяя эти моноидные значения с помощью функции mappend
, она сводит их все в одно моноидное значение. На данный момент функция может показаться несколько странной, но вы увидите, что её очень просто реализовать. И такой реализации достаточно, чтобы определить для нашего типа экземпляр класса Foldable
! Поэтому если мы просто реализуем функцию foldMap
для какого-либо типа, то получаем функции foldr
и foldl
для этого типа даром!
Вот как мы делаем экземпляр класса Foldable
для типа:
instance F.Foldable Tree where
foldMap f EmptyTree = mempty
foldMap f (Node x l r) = F.foldMap f l `mappend`
f x `mappend`
F.foldMap f r
Если нам предоставлена функция, которая принимает элемент нашего дерева и возвращает моноидное значение, то как превратить наше целое дерево в одно моноидное значение? Когда мы использовали функцию fmap
с нашим деревом, мы применяли функцию, отображая с её помощью узел, а затем рекурсивно отображали с помощью этой функции левое поддерево, а также правое поддерево. Здесь наша задача состоит не только в отображении с помощью функции, но также и в соединении значений в одно моноидное значение с использованием функции mappend
. Сначала мы рассматриваем случай с пустым деревом – печальным и одиноким деревцем, у которого нет никаких значений или поддеревьев. Оно не содержит значений, которые мы можем предоставить нашей функции, создающей моноид, поэтому мы просто говорим, что если наше дерево пусто, то моноидное значение, в которое оно будет превращено, равно значению mempty
.
Интервал:
Закладка: