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

Посмотрим на эту простую функцию:
multThree :: Int -> Int -> Int -> Int
multThree x y z = x * y * z
Что происходит, если мы вызываем multThree 3 5 9
или ((multThree 3) 5) 9
? Сначала значение 3
применяется к multThree
, так как они разделены пробелом. Это создаёт функцию, которая принимает один параметр и возвращает новую функцию, умножающую на 3
. Затем значение 5
применяется к новой функции, что даёт функцию, которая примет параметр и умножит его уже на 15
. Значение 9
применяется к этой функции, и получается результат 135
. Вы можете думать о функциях как о маленьких фабриках, которые берут какие-то материалы и что-то производят. Пользуясь такой аналогией, мы даём фабрике multThree
число 3
, и, вместо того чтобы выдать число, она возвращает нам фабрику немного поменьше. Эта новая фабрика получает число 5
и тоже выдаёт фабрику. Третья фабрика при получении числа 9
производит, наконец, результат — число 135
. Вспомним, что тип этой функции может быть записан так:
multThree :: Int -> (Int -> (Int -> Int))
Перед символом –>
пишется тип параметра функции; после записывается тип значения, которое функция вернёт. Таким образом, наша функция принимает параметр типа Int
и возвращает функцию типа Int -> (Int –> Int)
. Аналогичным образом эта новая функция принимает параметр типа Int
и возвращает функцию типа Int -> Int
. Наконец, функция принимает параметр типа Int
и возвращает значение того же типа Int
.
Рассмотрим пример создания новой функции путём вызова функции с недостаточным числом параметров:
ghci> let multTwoWithNine = multThree 9
ghci> multTwoWithNine 2 3
54
В этом примере выражение multThree 9
возвращает функцию, принимающую два параметра. Мы называем эту функцию multTwoWithNine
. Если при её вызове предоставить оба необходимых параметра, то она перемножит их между собой, а затем умножит произведение на 9
.
Вызывая функции не со всеми параметрами, мы создаём новые функции «на лету». Допустим, нужно создать функцию, которая принимает число и сравнивает его с константой 100
. Можно сделать это так:
compareWithHundred :: Int -> Ordering
compareWithHundred x = compare 100 x
Если мы вызовем функцию с 99
, она вернёт значение GT
. Довольно просто. Обратите внимание, что параметр x
находится с правой стороны в обеих частях определения. Теперь подумаем, что вернёт выражение compare 100
. Этот вызов вернёт функцию, которая принимает параметр и сравнивает его с константой 100
. Ага-а! Не этого ли мы хотели? Можно переписать функцию следующим образом:
compareWithHundred :: Int -> Ordering
compareWithHundred = compare 100
Объявление типа не изменилось, так как выражение compare 100
возвращает функцию. Функция compare
имеет тип (Ord a) => a –> (a –> Ordering)
. Когда мы применим её к 100, то получим функцию, принимающую целое число и возвращающую значение типа Ordering
.
Сечения
Инфиксные функции могут быть частично применены при помощи так называемых сечений . Для построения сечения инфиксной функции достаточно поместить её в круглые скобки и предоставить параметр только с одной стороны. Это создаст функцию, которая принимает один параметр и применяет его к стороне с пропущенным операндом. Вот донельзя простой пример:
divideByTen :: (Floating a) => a –> a
divideByTen = (/10)
Вызов, скажем, divideByTen 200
эквивалентен вызову 200 / 10
, равно как и (/10) 200
:
ghci> divideByTen 200
20.0
ghci> 200 / 10
20.0
ghci> (/10) 200
20.0
А вот функция, которая проверяет, находится ли переданный символ в верхнем регистре:
isUpperAlphanum :: Char –> Bool
isUpperAlphanum = (`elem` ['А'..'Я'])
Единственная особенность при использовании сечений – применение знака «минус». По определению сечений, (–4)
– это функция, которая вычитает четыре из переданного числа. В то же время для удобства (–4) означает «минус четыре». Если вы хотите создать функцию, которая вычитает четыре из своего аргумента, выполняйте частичное применение таким образом: (subtract 4)
.
Печать функций
До сих пор мы давали частично применённым функциям имена, после чего добавляли недостающие параметры, чтобы всё-таки посмотреть на результаты. Однако мы ни разу не попробовали напечатать сами функции. Попробуем? Что произойдёт, если мы попробуем выполнить multThree 3 4
в GHCi вместо привязки к имени с помощью ключевого слова let
либо передачи другой функции?
ghci> multThree 3 4
:1:0:
No instance for (Show (a –> a))
arising from a use of `print' at :1:0–12
Possible fix: add an instance declaration for (Show (a –> a))
In the expression: print it
In a 'do' expression: print it
GHCi сообщает нам, что выражение порождает функцию типа a –> a
, но он не знает, как вывести её на экран. Функции не имеют экземпляра класса Show
, так что мы не можем получить точное строковое представление функций. Когда мы вводим, скажем, 1 + 1
в терминале GHCi, он сначала вычисляет результат ( 2
), а затем вызывает функцию show
для 2
, чтобы получить текстовое представление этого числа. Текстовое представление 2
– это строка "2"
, которая и выводится на экран.
Интервал:
Закладка: