Миран Липовача - Изучай Haskell во имя добра!
- Название:Изучай Haskell во имя добра!
- Автор:
- Жанр:
- Издательство:ДМК Пресс
- Год:2012
- Город:Москва
- ISBN:978-5-94074-749-9
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Миран Липовача - Изучай Haskell во имя добра! краткое содержание
Язык Haskell имеет множество впечатляющих возможностей, но главное его свойство в том, что меняется не только способ написания кода, но и сам способ размышления о проблемах и возможных решениях. Этим Haskell действительно отличается от большинства языков программирования. С его помощью мир можно представить и описать нестандартным образом. И поскольку Haskell предлагает совершенно новые способы размышления о проблемах, изучение этого языка может изменить и стиль программирования на всех прочих.
Ещё одно необычное свойство Haskell состоит в том, что в этом языке придаётся особое значение рассуждениям о типах данных. Как следствие, вы помещаете больше внимания и меньше кода в ваши программы.
Вне зависимости от того, в каком направлении вы намерены двигаться, путешествуя в мире программирования, небольшой заход в страну Haskell себя оправдает. А если вы решите там остаться, то наверняка найдёте чем заняться и чему поучиться!
Эта книга поможет многим читателям найти свой путь к Haskell.
Отображения, монады, моноиды и другое! Всё сказано в названии: «Изучай Хаскель во имя добра!» – весёлый иллюстрированный самоучитель по этому сложному функциональному языку.
С помощью оригинальных рисунков автора, отсылке к поп-культуре, и, самое главное, благодаря полезным примерам кода, эта книга обучает основам функционального программирования так, как вы никогда не смогли бы себе представить.
Вы начнете изучение с простого материала: основы синтаксиса, рекурсия, типы и классы типов. Затем, когда вы преуспеете в основах, начнется настоящий мастер-класс от профессионала: вы изучите, как использовать аппликативные функторы, монады, застежки, и другие легендарные конструкции Хаскеля, о которых вы читали только в сказках.
Продираясь сквозь образные (и порой безумные) примеры автора, вы научитесь:
• Смеяться в лицо побочным эффектам, поскольку вы овладеете техниками чистого функционального программирования.
• Использовать волшебство «ленивости» Хаскеля для игры с бесконечными наборами данных.
• Организовывать свои программы, создавая собственные типы, классы типов и модули.
• Использовать элегантную систему ввода-вывода Хаскеля, чтобы делиться гениальностью ваших программ с окружающим миром.
Нет лучшего способа изучить этот мощный язык, чем чтение «Изучай Хаскель во имя добра!», кроме, разве что, поедания мозга его создателей. Миран Липовача (Miran Lipovača) изучает информатику в Любляне (Словения). Помимо его любви к Хаскелю, ему нравится заниматься боксом, играть на бас-гитаре и, конечно же, рисовать. У него есть увлечение танцующими скелетами и числом 71, а когда он проходит через автоматические двери, он притворяется, что на самом деле открывает их силой своей мысли.
Изучай Haskell во имя добра! - читать онлайн бесплатно ознакомительный отрывок
Интервал:
Закладка:
Аппликативный стиль
При использовании класса типов Applicative
мы можем последовательно задействовать несколько операторов <*>
в виде цепочки вызовов, что позволяет легко работать сразу с несколькими аппликативными значениями, а не только с одним. Взгляните, например, на это:
ghci> pure (+) <*> Just 3 <*> Just 5
Just 8
ghci> pure (+) <*> Just 3 <*> Nothing
Nothing
ghci> pure (+) <*> Nothing <*> Just 5
Nothing
Мы обернули оператор +
в аппликативное значение, а затем использовали оператор <*>
, чтобы вызвать его с двумя параметрами, оба из которых являются аппликативными значениями.

Давайте посмотрим, как это происходит, шаг за шагом. Оператор <*>
левоассоциативен; это значит, что
pure (+) <*> Just 3 <*> Just 5
то же самое, что и вот это:
(pure (+) <*> Just 3) <*> Just 5
Сначала оператор +
помещается в аппликативное значение – в данном случае значение типа Maybe
, которое содержит функцию. Итак, у нас есть pure (+)
, что, по сути, равно Just (+)
. Далее происходит вызов Just (+) <*> Just 3
. Его результатом является Just (3+)
. Это из-за частичного применения. Применение только значения 3
к оператору +
возвращает в результате функцию, которая принимает один параметр и добавляет к нему 3
. Наконец, выполняется Just (3+) <*> Just 5
, что в результате возвращает Just 8
.
Ну разве не здорово?! Аппликативные функторы и аппликативный стиль вычисления pure f <*> x <*> y <*>
… позволяют взять функцию, которая ожидает параметры, не являющиеся аппликативными значениями, и использовать эту функцию для работы с несколькими аппликативными значениями. Функция может принимать столько параметров, сколько мы захотим, потому что она всегда частично применяется шаг за шагом между вхождениями оператора <*>
.
Это становится ещё более удобным и очевидным, если мы примем во внимание тот факт, что выражение pure f <*> x
равно fmap f x
. Это один из законов аппликативных функторов, которые мы более подробно рассмотрим чуть позже; но давайте подумаем, как он применяется здесь. Функция pure
помещает значение в контекст по умолчанию. Если мы просто поместим функцию в контекст по умолчанию, а затем извлечём её и применим к значению внутри другого аппликативного функтора, это будет то же самое, что просто отобразить этот аппликативный функтор с помощью данной функции. Вместо записи pure f <*> x <*> y <*>
…, мы можем написать fmap
f
x
<*>
y
<*>
… Вот почему модуль Control.Applicative
экспортирует оператор, названный <$>
, который является просто синонимом функции fmap
в виде инфиксного оператора. Вот как он определён:
(<$>) :: (Functor f) => (a –> b) –> f a –> f b
f <$> x = fmap f x
ПРИМЕЧАНИЕ.Вспомните, что переменные типов не зависят от имён параметров или имён других значений. Здесь идентификатор f
в сигнатуре функции является переменной типа с ограничением класса, которое говорит, что любой конструктор типа, который заменяет f
, должен иметь экземпляр класса Functor
. Идентификатор f
в теле функции обозначает функцию, с помощью которой мы отображаем значение x
. Тот факт, что мы использовали f
для представления обеих вещей, не означает, что они представляют одну и ту же вещь.
При использовании оператора <$>
аппликативный стиль проявляет себя во всей красе, потому что теперь, если мы хотим применить функцию f
к трем аппликативным значениям, можно просто написать f <$> x <*> y <*> z
. Если бы параметры были обычными значениями, мы бы написали f x y z
.
Давайте подробнее рассмотрим, как это работает. Предположим, что мы хотим соединить значения Just "johntra"
и Just "volta"
в одну строку, находящуюся внутри функтора Maybe
. Сделать это вполне в наших силах!
ghci> (++) <$> Just "johntra" <*>
Just "volta" Just "johntravolta"
Прежде чем мы увидим, что происходит, сравните предыдущую строку со следующей:
ghci> (++) "johntra" "volta"
"johntravolta"
Чтобы использовать обычную функцию с аппликативным функтором, просто разбросайте вокруг несколько <$>
и <*>
, и функция будет работать с аппликативными значениями и возвращать аппликативное значение. Ну не здорово ли?
Возвратимся к нашему выражению (++) <$> Just "джонтра" <*> Just "волта"
: сначала оператор (++)
, который имеет тип (++) :: [a] – > [a] –> [a]
, отображает значение Just "джонтра"
. Это даёт в результате такое же значение, как Just ("джонтра"++)
, имеющее тип Maybe ([Char] –> [Char])
. Заметьте, как первый параметр оператора (++)
был «съеден» и идентификатор a
превратился в тип [Char]
! А теперь выполняется выражение Just ("джонтра"++) <*> Just "волта"
, которое извлекает функцию из Just
и отображает с её помощью значение Just "волта"
, что в результате даёт новое значение – Just "джонтраволта"
. Если бы одним из двух значений было значение Nothing
, результатом также было бы Nothing
.
Списки
Списки (на самом деле конструктор типа списка, []
) являются аппликативными функторами. Вот так сюрприз! Вот как []
является экземпляром класса Applicative
:
instance Applicative [] where
pure x = [x]
fs <*> xs = [f x | f <���– fs, x <���– xs]
Вспомните, что функция pure
принимает значение и помещает его в контекст по умолчанию. Другими словами, она помещает его в минимальный контекст, который всё ещё возвращает это значение. Минимальным контекстом для списков был бы пустой список, но пустой список означает отсутствие значения, поэтому он не может содержать в себе значение, к которому мы применили функцию pure
. Вот почему эта функция принимает значение и помещает его в одноэлементный список. Подобным образом минимальным контекстом для аппликативного функтора Maybe
было бы значение Nothing
– но оно означает отсутствие значения вместо самого значения, поэтому функция pure
в реализации экземпляра для типа Maybe
реализована как вызов конструктора данных Just
.
Вот функция pure
в действии:
ghci> pure "Эй" :: [String]
["Эй"]
ghci> pure "Эй" :: Maybe String
Just "Эй"
Что насчёт оператора <*>
? Если бы тип оператора <*>
ограничивался только списками, мы получили бы (<*>) :: [a –> b] –> [a] –> [b]
. Этот оператор реализован через генератор списков. Он должен каким-то образом извлечь функцию из своего левого параметра, а затем с её помощью отобразить правый. Но левый список может не содержать в себе функций или содержать одну либо несколько функций, а правый список также может содержать несколько значений. Вот почему мы используем генератор списков для извлечения из обоих списков. Мы применяем каждую возможную функцию из левого списка к каждому возможному значению из правого. Результирующий список содержит все возможные комбинации применения функции из левого списка к значению из правого.
Интервал:
Закладка: