Возможности Haskell - Haskell features

В этой статье описываются функции Haskell.

Примеры

Факториал

Простой пример, который часто используется для демонстрации синтаксиса функциональные языки это факториал функция для неотрицательных целых чисел, показанная в Haskell:

факториал :: Целое число -> Целое числофакториал 0 = 1факториал п = п * факториал (п-1)

Или одной строкой:

факториал п = если п > 1 тогда п * факториал (п-1) еще 1

Это описывает факториал как рекурсивную функцию с одним завершающим базовым случаем. Это похоже на описания факториалов в учебниках математики. Большая часть кода Haskell похожа на стандартный математическая запись в средствах и синтаксисе.

Первая строка факториальной функции описывает тип этой функции; Хотя это необязательно, это считается хорошим стилем[1] включить его. Это можно читать как функция факториал (факториал) имеет тип (::) от целого к целому (Целое -> Целое). То есть он принимает целое число в качестве аргумента и возвращает другое целое число. Тип определения выводится автоматически, если программист не предоставил аннотацию типа.

Вторая линия опирается на сопоставление с образцом, важная особенность Haskell. Обратите внимание, что параметры функции заключаются не в скобки, а через пробелы. Когда аргумент функции равен 0 (нулю), она вернет целое число 1 (один). Во всех остальных случаях пробуется третья строка. Это рекурсия, и снова выполняет функцию, пока не будет достигнут базовый вариант.

С использованием товар функции из Prelude, ряд небольших функций, аналогичных C с стандартная библиотека, и используя синтаксис Haskell для арифметических последовательностей, факториальная функция может быть выражена в Haskell следующим образом:

факториал п = товар [1..п]

Здесь [1..n] обозначает арифметическую последовательность 1, 2, …, п в виде списка. Использование функции Prelude enumFromTo, выражение [1..n] можно записать как enumFromTo 1 n, позволяя выразить факториальную функцию как

факториал п = товар (enumFromTo 1 п)

который, используя оператор композиции функции (выражается точкой в ​​Haskell), чтобы составить функцию произведения с карри функцию перечисления можно переписать на безточечный стиль:[2]

факториал = товар . enumFromTo 1

В интерпретаторе Hugs часто требуется определить функцию и использовать ее в той же строке, разделенной знаком куда или же позволять..в. Например, чтобы протестировать приведенные выше примеры и увидеть результат 120:

позволять { факториал п | п > 0 = п * факториал (п-1); факториал _ = 1 } в факториал 5

или же

факториал 5 куда факториал = товар . enumFromTo 1

Интерпретатор GHCi не имеет этого ограничения, и определения функций можно вводить в одной строке (с позволять синтаксис без в часть) и упоминается позже.

Более сложные примеры

Калькулятор

В исходном коде Haskell непосредственно ниже "::" можно читать как "имеет тип"; «a -> b» можно прочитать как «это функция от a до b». (Таким образом, Haskell "calc :: String -> [Float]" можно читать как "расчет имеет тип функции от строк до списков чисел с плавающей запятой ".) Во второй строке" calc = ... "знак равенства может читаться как" может быть "; таким образом, можно прочитать несколько строк с" calc = ... " как несколько возможных значений для расчетв зависимости от обстоятельств, указанных в каждой строке.

Простой Обратная польская запись калькулятор выражается с помощью функция высшего порядка складка чей аргумент ж определяется в куда пункт с использованием сопоставление с образцом и тип класс Читать:

расчет :: Нить -> [Плавать]расчет = складка ж [] . слова  куда     ж (Икс:у:zs) "+" = (у + Икс):zs    ж (Икс:у:zs) "-" = (у - Икс):zs    ж (Икс:у:zs) "*" = (у * Икс):zs    ж (Икс:у:zs) "/" = (у / Икс):zs    ж (Икс:у:zs) "КУВЫРОК" =  у:Икс:zs    ж хз у = читать у : хз

Пустой список - это начальное состояние, а ж интерпретирует одно слово за раз, либо как имя функции, беря два числа из заголовка списка и возвращая результат обратно, либо разбирая слово как число с плавающей запятой и добавив его в список.

Последовательность Фибоначчи

Следующее определение дает список Числа Фибоначчи в линейное время:

выдумки = 0 : 1 : zipWith (+) выдумки (хвост выдумки)

Бесконечный список создается Corecursion - последние значения списка вычисляются по запросу, начиная с первых двух пунктов 0 и 1. Это определение основывается на ленивая оценка, важная особенность программирования на Haskell. Для примера того, как развивается оценка, ниже показаны значения выдумки и хвостик после вычисления шести пунктов и показывает, как zipWith (+) произвел четыре предмета и приступает к производству следующего предмета:

fibs = 0: 1: 1: 2: 3: 5: ... + + + + + + хвост fibs = 1: 1: 2: 3: 5: ... = = = = = = zipWith ... = 1: 2: 3: 5: 8 : ... fibs = 0: 1: 1: 2: 3: 5: 8 : ...

Та же функция, написанная с использованием GHC понимание параллельного списка синтаксис (расширения GHC должны быть включены с помощью специального флага командной строки, здесь -XParallelListComp, или запустив исходный файл с {- # LANGUAGE ParallelListComp # -}):

выдумки = 0 : 1 : [ а+б | а <- выдумки | б <- хвост выдумки ]

или с обычным составить список:

выдумки = 0 : 1 : [ а+б | (а,б) <- застегивать выдумки (хвост выдумки) ]

или напрямую ссылаясь на себя:

выдумки = 0 : 1 : следующий выдумки куда следующий (а : т@(б:_)) = (а+б) : следующий т

С сохранный создание функция:

выдумки = следующий (0,1) куда следующий (а,б) = а : следующий (б, а+б)

или с разворачивать:

выдумки = разворачивать (\(а,б) -> Только (а, (б, а+б))) (0, 1)

или же сканл:

выдумки = 0 : сканл (+) 1 выдумки

Использование рекурсии данных с предопределенными Haskell комбинатор фиксированной точки:

выдумки = исправить (\хз -> 0 : 1 : zipWith (+) хз (хвост хз))   - zipWith версии     = исправить ((0:) . (1:) . (zipWith (+) <*> хвост))      - то же, что и выше, без точек     = исправить ((0:) . сканл (+) 1)                        - версия scanl

Факториал

Факториал, который мы видели ранее, может быть записан как последовательность функций:

факториал п = складной ((.) . (*)) я бы [1..п] $ 1- факториал 5 == ((1 *).) (((2 *).) (((3 *).) (((4 *).) (((5 *).) id)))) 1- == (1 *). (2 *). (3 *). (4 *). (5 *). id $ 1- == 1 * (2 * (3 * (4 * (5 * (id 1)))))факториал п = складной ((.) . (*)) (const 1) [1..п] $ ()- факториал 5 == ((1 *).) (((2 *).) (((3 *).) (((4 *).) (((5 *).) (const 1)) ))) ()- == (1 *). (2 *). (3 *). (4 *). (5 *). const 1 $ ()- == 1 * (2 * (3 * (4 * (5 * (const 1 ())))))факториал п = складной (($) . (*)) 1 [1..п] = складной ($) 1 $ карта (*) [1..п]- факториал 5 == ((1 *) $) (((2 *) $) (((3 *) $) (((4 *) $) (((5 *) $) 1))))--             == (1*) $ (2*) $ (3*) $ (4*) $ (5*) $ 1--             ==  1*  (  2*  (  3*  (  4*  (  5*    1 ))))

Еще примеры

Числа Хэмминга

Замечательно лаконичная функция, которая возвращает список Числа Хэмминга чтобы:

хамминг = 1 : карта (2*) хамминг `союз` карта (3*) хамминг                                  `союз` карта (5*) хамминг

Как и различные выдумки В решениях, показанных выше, используется corecursion для создания списка чисел по запросу, начиная с базового случая 1 и создавая новые элементы на основе предыдущей части списка.

Здесь функция союз используется как оператор, заключая его в обратные кавычки. Его дело статьи определяют, как это сливается два восходящих списка в один восходящий список без повторяющихся элементов, представляющих наборы как упорядоченные списки. Его функция-компаньон минус орудия установить разницу:

союз (Икс:хз) (у:ys) = дело сравнивать Икс у из    LT -> Икс : союз  хз (у:ys)      Эквалайзер -> Икс : союз  хз    ys      GT -> у : союз (Икс:хз) ys  союз  хз  []  = хз  союз  []  ys  = ys
минус (Икс:хз) (у:ys) = дело сравнивать Икс у из     LT -> Икс : минус  хз (у:ys)    Эквалайзер ->     минус  хз    ys     GT ->     минус (Икс:хз) ysминус  хз  _  = хз--

Для более эффективной работы можно генерировать только уникальные кратные. Поскольку дубликатов нет, удалять их не нужно:

гладкий235 = 1 : складной (\п s -> исправить $ mergeBy (<) s . карта (п*) . (1:)) [] [2,3,5]  куда    исправить ж = Икс  куда Икс = ж Икс         - комбинатор фиксированных точек с совместным использованием

Это использует более эффективную функцию слияние который не касается дубликатов (также используется в следующей следующей функции, Сортировка слиянием ):

mergeBy меньше хз ys = слияние хз ys  куда  слияние  хз     []  = хз   слияние  []     ys  = ys  слияние (Икс:хз) (у:ys) | меньше у Икс  = у : слияние (Икс:хз) ys                      | иначе = Икс : слияние хз (у:ys)

Каждая вертикальная полоса ( | ) начинает охранную оговорку с выражение охранника перед = знак и соответствующее определение после него, которое оценивается, если охранник истинен.

Сортировка слиянием

Вот снизу вверх Сортировка слиянием, определенный с помощью функция высшего порядка до того как:

mergesortBy меньше [] = []mergesortBy меньше хз = голова $      до того как (ноль . хвост) (попарно $ mergeBy меньше) [[Икс] | Икс <- хз]попарно ж (а:б:т) = ж а б : попарно ж тпопарно ж      т  = т

простые числа

Математическое определение простые числа можно почти дословно перевести на Haskell:

- «Целые числа больше 1, которые нельзя разделить на меньшее целое число больше 1»простые числа = [ п | п <- [2..], все ((> 0) . rem п) [2..(п-1)] ]

Это находит простые числа судебное отделение. Обратите внимание, что он не оптимизирован для повышения эффективности и имеет очень низкую производительность. Чуть быстрее (но все равно очень медленно)[3] это код от Дэвид Тернер:

простые числа = сито [2..]  куда          сито (п:хз) = п : сито [Икс | Икс <- хз, rem Икс п /= 0]

Намного быстрее работает оптимальный алгоритм пробного деления

простые числа = 2 : [ п | п <- [3..], все ((> 0) . rem п) $                      взять ((<= п) . (^2)) простые числа]

или неограниченный сито Эратосфена с отложенным поэтапным рассевом,[4]

простые числа = 2 : сито простые числа [3..]  куда             сито (п:пс) (охватывать (< п*п) -> (час, т)) =                    час ++ сито пс (минус т [п*п, п*п+п..])

или реализация комбинированного сита Ричард Берд,[5]

- "Целые числа больше 1 без составных чисел, которые- находятся путем перечисления кратных каждого простого числа "простые числа = 2 : минус [3..]               (складной (\(м:РС) р -> м : союз РС р) []                       [[п*п, п*п+п ..] | п <- простые числа])

или даже быстрее древовидная складчатость вариант[6] с почти оптимальной (для кода на основе списка) временной сложностью и очень низкой пространственной сложностью, достигаемой за счет телескопического многоступенчатого рекурсивного создания простых чисел:

простые числа = 2 : _Y ((3 :) . минус [5,7..] . _U                        . карта (\п -> [п*п, п*п+2*п..]))  куда    - комбинатор Y без совместного использования:    _Y грамм = грамм (_Y грамм)     - (г (г (г (г (...)))))    - большой союз ~ = nub.sort.concat    _U ((Икс:хз):т) = Икс : (союз хз . _U . попарно союз) т

Работая с массивами по сегментам между последовательными квадратами простых чисел, он

импорт Data.Arrayимпорт Data.List (хвосты, inits)пс = 2 : [п |    (р:q:_, px) <- (застегивать . хвосты . (2:) . карта (^2)) пс (inits пс),   (п,Истинный)    <- ассоциаты (аккумулятор (\_ _ -> Ложь) Истинный (р+1,q-1)                             [(м,()) | п <- px                                     , позволять s = div (р+п) п * п                                     , м <- [s,s+п..q-1]])]

Самый короткий код, вероятно, nubBy (((> 1).). gcd) [2 ..]. Это довольно медленно.

Синтаксис

Макет

Haskell позволяет отступ будет использоваться для обозначения начала нового объявления. Например, в куда пункт:

товар хз = толкать хз 1  куда    толкать []     а = а    толкать (Икс:хз) а = толкать хз (а*Икс)

Два уравнения для вложенная функция толкать выровнены по вертикали, что позволяет опустить разделитель с запятой. В Haskell отступы можно использовать в нескольких синтаксических конструкциях, включая делать, позволять, дело, учебный класспример.

Использование отступов для обозначения структуры программы возникает в Приземлиться с Я ПЛАВАЮ язык, где это называлосьвне игры. Позже это было принято Миранда, и Haskell принял аналогичную (но, скорее, более сложную) версию правила Миранды офф-сайд, которое называется «макет». Другие языки, в которых используется синтаксис с учетом пробелов, включают Python и F #.

Использование макета в Haskell не является обязательным. Например, функция товар выше также можно написать:

товар хз = толкать хз 1  куда { толкать [] а = а; толкать (Икс:хз) а = толкать хз (а*Икс) }

Явная открытая скобка после куда Ключевое слово указывает, что программист решил использовать явные точки с запятой для разделения деклараций, и что список деклараций будет завершен явной закрывающей скобкой. Одна из причин, по которой требуется поддержка явных ограничителей, заключается в том, что они упрощают автоматическую генерацию исходного кода Haskell.

Правило компоновки Haskell подвергалось критике за его сложность. В частности, определение гласит, что если синтаксический анализатор обнаруживает неполную ошибку во время обработки раздела макета, он должен попытаться вставить закрывающую скобку (правило «ошибки синтаксического анализа»). Реализация этого правила в традиционном разбор /лексический анализ сочетание требует двустороннего взаимодействия между синтаксическим анализатором и лексическим анализатором, тогда как в большинстве языков эти две фазы можно рассматривать независимо.

Вызов функций

Применение функции ж к значению Икс выражается просто f x.

Haskell отличает вызовы функций от инфиксных операторов синтаксически, но не семантически. Имена функций, состоящие из знаков препинания, могут использоваться в качестве операторов, как и другие имена функций, если они окружены обратными кавычками; Операторы and можно использовать в префиксной нотации, если они заключены в круглые скобки.

В этом примере показаны способы вызова функций:

  Добавить а б = а + б  десять1 = 5 + 5  десять2 = (+) 5 5  десять3 = Добавить 5 5  десять4 = 5 `Добавить` 5

Функции, которые определены как принимающие несколько параметров, всегда могут быть частично применены. Бинарные операторы можно частично применить с помощью раздел обозначение:

  десять5 = (+ 5) 5  десять6 = (5 +) 5    добавить пять = (5 +)  десять7 = добавить пять 5

Составьте список

Видеть Понимание списка # Обзор для примера Haskell.

Сопоставление с образцом

Сопоставление с образцом используется для сопоставления различных конструкторов алгебраических типов данных. Вот несколько функций, каждая из которых использует сопоставление с образцом для каждого из перечисленных ниже типов:

- Эта сигнатура типа говорит, что empty принимает список, содержащий любой тип, и возвращает Boolпустой :: [а] -> Boolпустой (Икс:хз) = Ложьпустой [] = Истинный- Вернет значение из Может быть, с заданным значением по умолчанию в случае, если Ничего не обнаруженоfromМожет быть :: а -> Может быть а -> аfromМожет быть Икс (Только у) = уfromМожет быть Икс Ничего  = Иксправильно :: Либо а б -> Boolправильно (Правильно _) = Истинныйправильно (Оставили _)  = ЛожьgetName :: Человек -> НитьgetName (Человек имя _ _) = имяgetSex :: Человек -> СексgetSex (Человек _ секс _) = сексgetAge :: Человек -> IntgetAge (Человек _ _ возраст) = возраст

Используя указанные выше функции вместе с карта функцию, мы можем применить их к каждому элементу списка, чтобы увидеть их результаты:

карта пустой [[1,2,3]],[],[2],[[1..]]- возвращает [False, True, False, False]карта (fromМожет быть 0) [Только 2,Ничего,Только 109238, Ничего]- возвращает [2,0,109238,0]карта правильно [Оставили "Привет", Правильно 6, Правильно 23, Оставили "Мир"]- возвращает [False, True, True, False]карта getName [Человек "Сара" женский 20, Человек "Алекс" Мужской 20, Том]- возвращает ["Сара", "Алекс", "Том"], используя определение для тома выше
  • Абстрактные типы
  • Списки

Кортежи

Кортежи в haskell можно использовать для хранения фиксированного количества элементов. Они используются для группировки фрагментов данных разных типов:

учетная запись :: (Нить, Целое число, Двойной) - Тип тройки, представляющей                                      - имя, баланс и процентная ставкаучетная запись = ("Джон Смит",102894,5.25)

Кортежи обычно используются в функциях zip * для размещения смежных элементов в отдельных списках вместе в кортежах (от zip4 до zip7 предоставляются в модуле Data.List):

- Определение функции почтового индекса. Другие функции zip * определяются аналогичнозастегивать :: [Икс] -> [у] -> [(Икс,у)]застегивать (Икс:хз) (у:ys) = (Икс,у) : застегивать хз ysзастегивать _      _      = []застегивать [1..5] "Привет"- возвращает [(1, 'h'), (2, 'e'), (3, 'l'), (4, 'l'), (5, 'o')]- и имеет тип [(Integer, Char)]zip3 [1..5] "Привет" [Ложь, Истинный, Ложь, Ложь, Истинный]- возвращает [(1, 'h', False), (2, 'e', ​​True), (3, 'l', False), (4, 'l', False), (5, 'o' ,Истинный)]- и имеет тип [(Integer, Char, Bool)]

В компиляторе GHC кортежи определяются размером от 2 до 62 элементов.

Пространства имён

в #More_complex_examples раздел выше, расчет используется в двух смыслах, показывая, что существует пространство имен класса типа Haskell, а также пространство имен для значений:

  1. Haskell тип класс за расчет. В домен и классифицировать может быть явно обозначен в классе типов Haskell.
  2. значение Haskell, формула или выражение для расчет.

Классы типов и полиморфизм

Алгебраические типы данных

Алгебраические типы данных широко используются в Haskell. Некоторые примеры из них - встроенный список, Может быть и Либо типы:

- Список a ([a]) является либо a consed (:) в другой список a, либо пустым списком ([])данные [а] = а : [а] | []- Что-то типа Maybe a - это либо Just something, либо Nothingданные Может быть а = Только а | Ничего- Что-то типа Либо atype btype является либо Left atype, либо Right btypeданные Либо а б = Оставили а | Правильно б

Пользователи языка также могут определять свои собственные абстрактные типы данных. Пример ADT, используемого для представления имени, пола и возраста человека, может выглядеть так:

данные Секс = Мужской | женскийданные Человек = Человек Нить Секс Int - Обратите внимание, что Person одновременно является конструктором и типом- Пример создания чего-то типа PersonТом :: ЧеловекТом = Человек "Том" Мужской 27

Система типов

  • Типовые классы
  • Тип по умолчанию
  • Перегруженные литералы
  • Высший родственный полиморфизм
  • Классы многопараметрических типов
  • Функциональные зависимости

Монады и ввод / вывод

ST монада

Монада ST позволяет программистам писать императивные алгоритмы на Haskell, используя изменяемые переменные (STRef) и изменяемые массивы (STArrays и STUArrays). Преимущество монады ST заключается в том, что она позволяет программистам писать код, который имеет внутренние побочные эффекты, такие как деструктивное обновление изменяемых переменных и массивов, при этом эти эффекты содержатся внутри монады. Результатом этого является то, что функции, написанные с использованием монады ST, кажутся полностью чистыми для остальной части программы. Это позволяет программистам создавать императивный код там, где нецелесообразно писать функциональный код, сохраняя при этом всю безопасность, которую обеспечивает чистый код.

Вот пример программы (взятой со страницы вики Haskell на ST монада ), который берет список чисел и суммирует их, используя изменяемую переменную:

импорт Control.Monad.STимпорт Data.STRefимпорт Control.MonadsumST :: Num а => [а] -> аsumST хз = runST $ делать            - runST берет код ST с сохранением состояния и делает его чистым.    подведены <- newSTRef 0         - Создайте STRef (изменяемую переменную)    форма_ хз $ \Икс -> делать          - Для каждого элемента списка аргументов хз ..        modifySTRef подведены (+Икс)  - добавьте его к тому, что у нас в n.    readSTRef подведены             - прочитать значение n, которое будет возвращено функцией runST выше.

Монада СТМ

Монада STM - это реализация Программная транзакционная память в Haskell. Он реализован в компиляторе GHC и позволяет изменять изменяемые переменные в сделки.

Стрелки

  • Аппликативные функторы
  • Стрелки

Поскольку Haskell - это чисто функциональный язык, функции не могут иметь побочных эффектов. Поскольку он не является строгим, он также не имеет четко определенного порядка оценки. Это проблема для реальных программ, которые, помимо прочего, должны взаимодействовать с окружающей средой. Haskell решает эту проблему с помощью монадические типы которые используют систему типов для обеспечения правильной последовательности императивных конструкций. Типичным примером является ввод-вывод, но монады полезны для многих других целей, включая изменяемое состояние, параллелизм и транзакционную память, обработку исключений и распространение ошибок.

Haskell предоставляет специальный синтаксис для монадических выражений, так что программы с побочными эффектами могут быть написаны в стиле, аналогичном текущим императивным языкам программирования; нет знания о математика за монадическим вводом-выводом для этого требуется. Следующая программа считывает имя из командной строки и выводит приветственное сообщение:

главный = делать putStrLn "Как твое имя?"          имя <- getLine          putStr ("Привет, " ++ имя ++ "! п")

Нотация do упрощает работу с монадами. Это do-выражение эквивалентно, но (возможно) его легче написать и понять, чем обессахаренный версия, использующая монадические операторы напрямую:

главный = putStrLn "Как твое имя?" >> getLine >>= \ имя -> putStr ("Привет, " ++ имя ++ "! п")
Смотрите также wikibooks: Transwiki: Список программ hello world # Haskell для другого примера, который печатает текст.

Параллелизм

Само определение языка Haskell не включаетпараллелизм или же параллелизм, хотя GHC поддерживает оба.

Параллельный Haskell - это расширение Haskell, обеспечивающее поддержку потоков и синхронизации.[7] Реализация Concurrent Haskell в GHC основана на мультиплексировании легких потоков Haskell на несколько тяжелых потоков ОС,[8] так что параллельные программы Haskell выполняются параллельно на мультипроцессор. Среда выполнения может поддерживать миллионы одновременных потоков.[9]

Реализация GHC использует динамический пул потоков ОС, позволяя потоку Haskell выполнять блокирующий системный вызов, не блокируя другие запущенные потоки Haskell.[10] Следовательно, легкие потоки Haskell обладают характеристиками тяжелых потоков ОС, и программист не знает деталей реализации.

Недавно,[когда? ] В Concurrent Haskell добавлена ​​поддержка Программная транзакционная память (STM), который представляет собой абстракцию параллелизма, в которой составные операции с общими данными выполняются атомарно, как транзакции.[11] Реализация STM GHC - единственная реализация STM на сегодняшний день, которая обеспечивает статическую гарантию времени компиляции, предотвращающую выполнение нетранзакционных операций внутри транзакции. Библиотека Haskell STM также предоставляет две операции, которых нет в других STM: повторить попытку и orElse, которые вместе позволяют определять операции блокировки в модульная и сборная мода.

Рекомендации

  1. ^ HaskellWiki: Типовые подписи как хороший стиль
  2. ^ HaskellWiki: Pointfree
  3. ^ «Простые числа - HaskellWiki». www.haskell.org.
  4. ^ «Простые числа - HaskellWiki». www.haskell.org.
  5. ^ О'Нил, Мелисса Э., "Подлинное сито Эратосфена", Журнал функционального программирования, опубликованный в Интернете издательством Cambridge University Press, 9 октября 2008 г. Дои:10.1017 / S0956796808007004 С. 10, 11.
  6. ^ «Простые числа - HaskellWiki». www.haskell.org.
  7. ^ Саймон Пейтон Джонс, Эндрю Гордон и Сигбьорн Финн. Параллельный Haskell. Симпозиум ACM SIGPLAN-SIGACT по принципам языков программирования (PoPL). 1996. (Некоторые разделы устарели по отношению к текущей реализации.)
  8. ^ Поддержка среды выполнения для многоядерного Haskell В архиве 2010-07-05 в Wayback Machine (Саймон Марлоу, Саймон Пейтон Джонс, Сатнам Сингх) ICFP '09: Материалы 14-й международной конференции ACM SIGPLAN по функциональному программированию, Эдинбург, Шотландия, август 2009 г.
  9. ^ "DEFUN 2009: многоядерное программирование на Haskell прямо сейчас!". 5 сентября 2009 г.
  10. ^ Расширение интерфейса внешних функций Haskell с помощью параллелизма В архиве 2010-07-03 на Wayback Machine (Саймон Марлоу, Саймон Пейтон Джонс, Вольфганг Таллер) Труды семинара ACM SIGPLAN по Haskell, страницы 57-68, Snowbird, Юта, США, сентябрь 2004 г.
  11. ^ Харрис, Тим; Марлоу, Саймон; Пейтон-Джонс, Саймон; Херлихи, Морис (2005). «Транзакции составной памяти». Материалы десятого симпозиума ACM SIGPLAN по принципам и практике параллельного программирования. CiteSeerX  10.1.1.67.3686.