Парадигма | Многопарадигма : функциональный, императивный, модульный |
---|---|
Семейство | ML |
Впервые появилось | 1983; 37 лет назад (1983 г.) |
Стабильный выпуск | Standard ML '97 / 1997; 23 года назад (1997 г.) |
Дисциплина печати | Предполагаемый, статический, сильный |
Расширения имен файлов | .sml |
Веб-сайт | sml- family.org |
Основные реализации | |
SML / NJ, MLton | |
Диалекты | |
Алиса, Concurrent ML, Зависимый ML | |
Под влиянием | |
ML, Хоуп, Паскаль | |
Под влиянием | |
Вяз, F#, F*, Haskell, OCaml, Python, Rust, Scala |
Standard ML (SML ) - универсальный, модульный, функциональный язык программирования с проверкой типа во время компиляции и выводом типа. Он популярен среди разработчиков компиляторов и исследователей языков программирования, а также при разработке средств доказательства теорем.
SML - это современный диалект ML, язык программирования, используемый в проекте доказательства теорем Logic for Computable Functions (LCF). Он отличается от широко используемых языков тем, что имеет формальную спецификацию, представленную как правила типизации и операционная семантика в The Definition of Standard ML.
Standard ML - это функциональный язык программирования с некоторыми нечистыми функциями. Программы, написанные на стандартном ML, состоят из выражений для оценки, в отличие от операторов или команд, хотя некоторые выражения возвращают тривиальное «единичное» значение и оцениваются только по их побочным эффектам.
Как и все языки функционального программирования, ключевой особенностью Standard ML является функция , которая используется для абстракции. Например, функция factorial может быть выражена как:
fun factorial n = if n = 0 then 1 else n * factorial (n - 1)
Стандартный компилятор ML необходим для вывода статический тип int ->int
этой функции без аннотаций пользовательского типа. То есть, он должен вывести, что n используется только с целочисленными выражениями и, следовательно, само должно быть целым числом, и что все выражения, производящие значение, в функции возвращают целые числа.
Та же самая функция может быть выражена с заменой условного выражения if-then-else последовательностью шаблонов факториальной функции, оцениваемых для определенных значений, разделенных символом '|', которые проверяются один за другим в порядок записи до тех пор, пока не будет найдено совпадение:
fun factorial 0 = 1 | факториал n = n * факториал (n - 1)
Это можно переписать, используя оператор case следующим образом:
val rec factorial = fn n =>case n of 0 =>1 | n =>n * factorial (n - 1)
или итеративно:
fun factorialIT value = let val flag = ref value и i = ref 1 in while! flag <>0 do (i: =! i *! flag; flag: =! flag - 1); ! я конец;
или как лямбда-функция :
val rec factorial = fn 0 =>1 | n =>n * factorial (n - 1)
Здесь ключевое слово val
вводит привязку идентификатора к значению, fn
вводит определение анонимная функция и case
вводят последовательность шаблонов и соответствующих выражений.
Используя локальную функцию, эту функцию можно переписать в более эффективном стиле хвостовой рекурсии.
fun factorial n = let fun lp (0, acc) = acc | lp (m, acc) = lp (m-1, m * acc) in lp (n, 1) end
(Значение выражения let - это значение выражения между в и end .) Инкапсуляция сохраняющего инвариант хвостово-рекурсивного жесткого цикла с одним или несколькими параметрами аккумулятора внутри внешней функции без инвариантов, как показано здесь, является распространенной идиомой в Стандартный ML и очень часто появляется в коде SML.
Синонимы типов определяются с помощью ключевого слова type . Вот синоним типа для точек на плоскости и функций, вычисляющих расстояния между двумя точками и площадь треугольника с заданными углами согласно формуле Герона. (Эти определения будут использоваться в последующих примерах).
type loc = real * real fun dist ((x0, y0), (x1, y1)) = let val dx = x1 - x0 val dy = y1 - y0 в Math.sqrt (dx * dx + dy * dy) end fun heron (a, b, c) = let val ab = dist (a, b) val bc = dist (b, c) val ac = dist (a, c) val perim = ab + bc + ac val s = perim / 2.0 в Math.sqrt (s * (s - ab) * (s - bc) * (s - ac)) end
Стандартный ML обеспечивает надежную поддержку алгебраические типы данных (сокращенно ADT). Тип данных ML можно рассматривать как непересекающееся объединение кортежей (или «сумму произведений»). Их легко определить и с ними легко программировать, в значительной степени из-за соответствия шаблонов Standard ML, а также проверки полноты шаблонов и проверки избыточности шаблонов в большинстве реализаций Standard ML.
Тип данных определяется ключевым словом datatype, как в
datatype shape = Circle of loc * real (* center and radius *) | Квадрат loc * real (* верхний левый угол и длина стороны; с выравниванием по оси *) | Треугольник loc * loc * loc (* corners *)
(См. Выше определение loc
.) Примечание: для определения рекурсивных конструкторов необходимы типы данных, а не синонимы типов. (В данном примере это не проблема.)
Порядок имеет значение при сопоставлении с образцом: сначала пробуются образцы, которые сначала вводятся в текстовом виде. Сопоставление с образцом может быть синтаксически встроено в определения функции следующим образом:
область развлечений (Круг (_, r)) = 3,14 * r * r | площадь (Квадрат (_, s)) = s * s | area (Triangle (a, b, c)) = heron (a, b, c) (* см. выше *)
Обратите внимание, что подкомпоненты, значения которых не нужны в конкретном вычислении, опускаются с подчеркиванием или так называемыми шаблоны подстановки.
Определение так называемой функции стиля "клаузальной формы", в которой образцы появляются сразу после имени функции, является просто синтаксическим сахаром для
форма области развлечения = форма формы круга ( _, r) =>3,14 * r * r | Квадрат (_, s) =>s * s | Треугольник (a, b, c) =>heron (a, b, c)
Проверка полноты шаблона гарантирует, что каждый случай типа данных учтен, и выдаст предупреждение, если нет. Следующий шаблон неисчерпаем:
центр развлечений (Круг (c, _)) = c | center (Square ((x, y), s)) = (x + s / 2.0, y + s / 2.0)
Нет шаблона для случая Triangle
в center
функция. Компилятор выдаст предупреждение о том, что шаблон неисчерпаем, и если во время выполнения этой функции будет передан треугольник , возникнет исключение
Match
.
Набор предложений в следующем определении функции является исчерпывающим и не избыточным:
fun hasCorners (Circle _) = false | hasCorners _ = true
Если элемент управления проходит мимо первого шаблона (Круг
), мы знаем, что значение должно быть либо Квадратом
, либо Треугольником
. В любом из этих случаев мы знаем, что у фигуры есть углы, поэтому мы можем вернуть true
, не различая, в каком случае мы находимся.
Шаблон во втором предложении следующего (бессмысленно) функция избыточна:
fun f (Circle ((x, y), r)) = x + y | f (Круг _) = 1.0 | f _ = 0.0
Любое значение, которое соответствует шаблону во втором предложении, также будет соответствовать шаблону в первом предложении, поэтому второе предложение недостижимо. Следовательно, это определение в целом демонстрирует избыточность и вызывает предупреждение во время компиляции.
Программисты на C могут использовать помеченные объединения, отправляя значения тегов, чтобы выполнить то, что ML выполняет с типами данных и сопоставлением с образцом. Тем не менее, хотя программа C, снабженная соответствующими проверками, будет в некотором смысле столь же надежна, как соответствующая программа ML, эти проверки обязательно будут динамическими; ML предоставляет набор статических проверок, которые дают программисту высокую степень уверенности в правильности программы во время компиляции.
Обратите внимание, что в объектно-ориентированных языках программирования, таких как Java, несвязное объединение может быть выражено путем проектирования иерархий классов. Однако, в отличие от иерархий классов, ADT закрыты. Это делает ADT расширяемым способом, ортогональным расширяемости иерархий классов. Иерархии классов могут быть расширены новыми подклассами, но не новыми методами, в то время как ADT могут быть расширены для обеспечения нового поведения для всех существующих конструкторов, но не позволяют определять новые конструкторы.
Функции могут использовать функции в качестве аргументов:
fun applyToBoth fxy = (fx, fy)
Функции могут создавать функции как возвращаемые значения:
fun constantFn k = let fun const something = k in const end
(альтернативно)
fun constantFn k = (fn something =>k)
Функции также могут потреблять и создавать функции:
fun compose ( f, g) = let fun hx = f (gx) in h end
(альтернативно)
fun compose (f, g) = (fn x =>f (gx))
Функция List.map
из базовой библиотеки - одна из наиболее часто используемых функций высшего порядка в Standard ML:
fun map _ = | map f (x :: xs) = fx :: map f xs
(Более эффективная реализация map
будет определять хвостовой рекурсивный внутренний цикл следующим образом :)
fun map f xs = let fun m (, acc) = List.rev acc | m (x :: xs, acc) = m (xs, fx :: acc) in m (xs,) end
Исключения возникают с помощью ключевого слова raise
, и обрабатывается с помощью конструкций дескриптора
сопоставления с образцом.
исключение Undefined fun max [x] = x | max (x :: xs) = let val m = max xs in if x>m then x else m end | max = поднять Undefined fun main xs = let val msg = (Int.toString (max xs)) handle Undefined =>«пустой список... нет максимума!» in print (msg ^ "\ n") end
Для реализации можно использовать систему исключений, метод оптимизации, подходящий для функций, подобных следующим.
исключение Zero fun listProd ns = let fun p = 1 | p (0 :: _) = поднять ноль | p (h :: t) = h * pt in (p ns) handle Zero =>0 end
Когда исключение Zero
возникает в случае 0, управление оставляет функцию p
вместе. Рассмотрим альтернативу: значение 0 будет возвращено в самый последний ожидающий кадр, оно будет умножено на локальное значение h
, результирующее значение (неизбежно 0) будет возвращено по очереди следующему ожидающему кадру. рама и так далее. Возникновение исключения позволяет обойти контроль над всей цепочкой кадров и избежать связанных вычислений. Следует отметить, что такую же оптимизацию можно было бы получить, используя хвостовую рекурсию для этого примера.
Standard ML имеет расширенную модульную систему, позволяющую разлагать программы на иерархически организованные структуры логически связанных объявлений типов и значений. Модули SML обеспечивают не только управление пространством имен, но также абстракцию в том смысле, что они позволяют программистам определять абстрактные типы данных.
Систему модулей SML составляют три основные синтаксические конструкции: структуры, сигнатуры и функторы. Структура - это модуль; он состоит из набора типов, исключений, значений и структур (называемых подструктурами), упакованных вместе в логическую единицу. Сигнатура - это интерфейс, обычно рассматриваемый как тип для структуры: он определяет имена всех сущностей, предоставляемых структурой, а также арностей компонентов типа, типы компонентов значения и подписи для подструктур. Определения компонентов типа можно экспортировать или не экспортировать; Компоненты типа, определения которых скрыты, являются абстрактными типами. Наконец, функтор - это функция от структур к структурам; то есть функтор принимает один или несколько аргументов, которые обычно являются структурами данной сигнатуры, и в качестве своего результата создает структуру. Функторы используются для реализации общих структур данных и алгоритмов.
Например, подпись для структуры данных очереди может быть:
подпись QUEUE = sig type 'исключение очереди QueueError val empty:' значение очереди isEmpty: 'очередь ->bool val singleton: 'a ->' a queue val insert: 'a *' a queue ->'a queue val peek:' a queue ->'a val remove:' a queue ->'a *' a queue end
Эта подпись описывает модуль, который предоставляет параметризованный тип queue
очередей, исключение с именем QueueError
и шесть значений (пять из которых являются функциями), обеспечивающими основные операции по очередям. Теперь можно реализовать структуру данных очереди, написав структуру с такой сигнатурой:
структура TwoListQueue:>QUEUE = struct type 'a queue =' a list * 'исключение списка QueueError val empty = (,) fun isEmpty (,) = истина | isEmpty _ = false весело singleton a = (, [a]) fun insert (a, (,)) = (, [a]) | insert (a, (ins, out)) = (a :: ins, out) fun peek (_,) = поднять QueueError | peek (ins, a :: out) = весело remove (_,) = поднять QueueError | remove (ins, [a]) = (a, (, rev ins)) | remove (ins, a :: out) = (a, (ins, out)) end
Это определение объявляет, что TwoListQueue
является реализацией сигнатуры QUEUE
. Кроме того, непрозрачное приписывание (обозначается :>
) указывает, что любые компоненты типа, определения которых не указаны в сигнатуре (т. Е. очередь
), должны рассматриваться как абстрактные, что означает, что определение очереди как пары списков не видно вне модуля. Тело структуры обеспечивает привязки для всех компонентов, перечисленных в подписи.
Чтобы использовать структуру, можно получить доступ к ее элементам типа и значения, используя "точечную нотацию". Например, очередь строк будет иметь тип string TwoListQueue.queue
, пустая очередь - TwoListQueue.empty
, а для удаления первого элемента из очереди будет назван q
можно было бы написать TwoListQueue.remove q
.
Один из популярных алгоритмов поиска в ширину деревьев использует очереди. Здесь мы представляем версию этого алгоритма, параметризованную для абстрактной структуры очереди:
функтор BFS (структура Q: QUEUE) = (* после Okasaki, ICFP, 2000 *) struct datatype 'a tree = E | T of 'a *' a tree * 'a tree fun bfsQ (q:' a tree Q.queue): 'a list = if Q.isEmpty q then else let val (t, q') = Q.remove q in случай t из E =>bfsQ q '| T (x, l, r) =>let val q '' = Q.insert (r, Q.insert (l, q ')) в x :: bfsQ q' 'end end fun bfs t = bfsQ (Q. singleton t) end
Обратите внимание, что внутри структуры BFS
программа не имеет доступа к конкретному представлению очереди в игре. Более конкретно, программа не может, скажем, выбрать первый список в представлении очереди с двумя списками, если это действительно используется представление. Этот механизм абстракции данных делает код в ширину действительно независимым от выбора представления очереди. Это вообще желательно; в данном случае структура очереди может безопасно поддерживать любой из различных логических инвариантов, от которых зависит ее правильность за пуленепробиваемой стеной абстракции.
Базовая библиотека SML стандартизирована и поставляется с большинством реализаций. Он предоставляет модули для деревьев, массивов и других структур данных, а также интерфейсы ввода / вывода и системы.
Для числовых вычислений существует модуль Matrix (но в настоящее время он не работает), https://www.cs.cmu.edu/afs/cs/project/ pscico / pscico / src / matrix / README.html.
Для графики cairo-sml - это интерфейс с открытым исходным кодом для графической библиотеки Cairo.
Для машинного обучения существует библиотека графических моделей.
Фрагменты кода SML легче всего изучать, вводя их в «верхний уровень», также известный как цикл чтения-оценки-печати или REPL. Это интерактивный сеанс, который печатает предполагаемые типы результирующих или определенных выражений. Многие реализации SML предоставляют интерактивный REPL, включая SML / NJ :
$ sml Standard ML of New Jersey v110.52 [построено: 21 января, пятница, 16:42:10 2005] -
После этого можно ввести код в ответ на приглашение «-». Например, чтобы вычислить 1 + 2 * 3:
- 1 + 2 * 3; val it = 7: int
Верхний уровень определяет тип выражения как «int» и возвращает результат «7».
Следующая программа, "hello.sml":
print "Привет, мир! \ N";
можно скомпилировать с помощью MLton:
$ mlton hello.sml
и запустить:
$./hello Hello world!
Сортировка вставкой для списков целых чисел (по возрастанию) кратко выражается следующим образом:
fun ins (n,) = [n] | ins (n, ns as h :: t) = if (nЭто можно сделать полиморфным, абстрагируясь от оператора упорядочивания. Здесь мы используем символическое имя
<<
для этого оператора.fun ins '<< (num, nums) = let fun i (n,) = [n] | i (n, ns as h::t) = if <<(n,h) then n::ns else h::i(n,t) in i (num, nums) end fun insertionSort' << = List.foldr (ins' <<)Тип
InsertSort'
-('a *' a ->bool) ->('список) ->(' список)
.Пример вызова это:
- InsertSort 'op < [3, 1, 4]; val it = [1,3,4] : int listMergesort
Здесь классический алгоритм сортировки слиянием реализован в трех функциях: split, merge и mergesort.
Функция
split
реализована с помощью локальной функции с именемloop
, которая имеет два дополнительных параметра. Локальная функцияloop
написана в стиле хвостовой рекурсии ; как таковая его можно эффективно скомпилировать. Эта функция использует синтаксис сопоставления с образцом SML, чтобы различать случаи непустого списка (x :: xs
) и пустого списка (). Для стабильности входной список
ns
перед передачей вloop
.переворачивается (* Разделить список на две почти половинки, возвращается как пара. * «Половинки» будут она должна быть того же размера *, иначе первый будет содержать на один элемент больше, чем второй. * Выполняется за время O (n), где n = | xs |. *) локальный цикл веселья (x :: y :: zs, xs, ys) = цикл (zs, x :: xs, y :: ys) | цикл (x ::, xs, ys) = (x :: xs, ys) | loop (, xs, ys) = (xs, ys) in fun split ns = loop (List.rev ns,) endСинтаксис local-in-end можно заменить синтаксисом let-in-end, что даст эквивалентное определение:
fun split ns = let fun loop (x :: y :: zs, xs, ys) = loop (zs, x :: xs, y :: ys) | цикл (x ::, xs, ys) = (x :: xs, ys) | loop (, xs, ys) = (xs, ys) in loop (List.rev ns,) endКак и в случае с split, слияние также использует цикл локальной функции для повышения эффективности. Внутренний цикл
определяется в терминах случаев: когда передаются два непустых списка, когда передается один непустой список и когда передаются два пустых списка. Обратите внимание на использование символа подчеркивания (
_
) в качестве шаблона подстановки.Эта функция объединяет два «восходящих» списка в один восходящий список. Обратите внимание, как аккумулятор
out
строится «в обратном порядке», а затем меняет его значение наList.rev
перед возвращением. Это распространенный метод - составьте список в обратном порядке, а затем переверните его перед возвратом. В SML списки представлены в виде связанного списка conses, и поэтому эффективно добавлять элемент в список, но неэффективно добавлять элемент в список. Дополнительный проход по списку представляет собой операцию с линейным временем, поэтому, хотя этот метод требует большего времени настенных часов, асимптотика не хуже.(* Объединить два упорядоченных списка, используя порядок lt. * Pre: указанные списки xs и ys уже должны быть упорядочены на lt. * Выполняется за время O (n), где n = | xs | + | ys |. *) fun merge lt (xs, ys) = let fun loop (out, left as x :: xs, right as y :: ys) = if lt (x, y) then loop (x :: out, xs, right) else loop (y :: out, left, ys) | loop (out, x :: xs,) = loop (x :: out, xs,) | цикл (выход, y :: ys) = цикл (y :: out, ys) | loop (out,) = List.rev out in loop (, xs, ys) endОсновная функция.
(* Сортировать список в соответствии с заданной операцией упорядочения lt. * Выполняется за O (n log n) раз, где n = | xs |. *) Fun mergesort lt xs = let val merge '= merge lt fun ms = | ms [x] = [x] | ms xs = let val (left, right) = split xs in merge '(ms left, ms right) end in ms xs endТакже обратите внимание, что в коде не упоминаются типы переменных, за исключением :: и синтаксис, обозначающий списки. Этот код будет сортировать списки любого типа, если можно определить последовательную функцию упорядочивания lt. Используя вывод типа Хиндли – Милнера, компилятор может вывести типы всех переменных, даже сложных типов, таких как тип функции lt.
Quicksort
Quicksort можно выразить следующим образом. Эта универсальная быстрая сортировка использует оператор порядка
<<
.fun quicksort << xs = let fun qs = | qs [x] = [x] | qs (p::xs) = let val (less, more) = List.partition (fn x =><< (x, p)) xs in qs less @ p :: qs more end in qs xs endНаписание языкового интерпретатора
Обратите внимание на относительную легкость, с которой язык малых выражений определяется и обрабатывается.
исключение Тип данных ошибки ty = IntTy | Тип данных BoolTy exp = True | Ложь | Int из int | Не эксп | Добавить exp * exp | Если exp * exp * exp fun typeOf (True) = BoolTy | typeOf (False) = BoolTy | typeOf (Int _) = IntTy | typeOf (Not e) = если typeOf e = BoolTy, то BoolTy иначе вызывает Err | typeOf (Add (e1, e2)) = if (typeOf e1 = IntTy) andalso (typeOf e2 = IntTy), то IntTy иначе вызывает Err | typeOf (If (e1, e2, e3)) = if typeOf e1 <>BoolTy, тогда поднять Err else if typeOf e2 <>typeOf e3 затем поднять Err else typeOf e2 fun eval (True) = True | eval (False) = False | eval (Int n) = Int n | eval (Not e) = (case eval e of True =>False | False =>True | _ =>raise Fail "проверка типов нарушена") | eval (Add (e1, e2)) = let val (Int n1) = eval e1 val (Int n2) = eval e2 в Int (n1 + n2) end | eval (If (e1, e2, e3)) = if eval e1 = True, затем eval e2 else eval e3 fun chkEval e = (ignore (typeOf e); eval e) (* вызовет Err при ошибке типа *)Пример использования на правильно набранных и неверно набранных примерах:
- val e1 = Add (Int (1), Int (2)); (* Правильно напечатано *) val e1 = Add (Int 1, Int 2): exp - chkEval e1; val it = Int 3: exp - val e2 = Add (Int (1), True); (* Неправильно введено *) val e2 = Add (Int 1, True): exp - chkEval e2; неперехваченное исключение ErrФакториальная функция произвольной точности (библиотеки)
В SML модуль IntInf обеспечивает целочисленную арифметику произвольной точности. Более того, целочисленные литералы могут использоваться как целые числа произвольной точности, при этом программисту ничего не нужно делать.
Следующая программа "fact.sml" реализует факториал произвольной точности и печатает факториал 120:
интересный факт n: IntInf.int = если n = 0, то 1 else n * fact ( п - 1) вал () = печать (IntInf.toString (факт 120) ^ "\ п")и может быть собран и работать с:
$ mlton fact.sml $./fact 66895029134491270575881180540903725867527463331380298102956713523016335 57244962989366874165271984981308157637893214090552534408589408121859898 481114389650005964960521256960000000000000000000000000000Числовая производная (функции высшего порядка)
Поскольку SML - это язык функционального программирования, легко создавать и передавать функции в программах SML. Эта возможность имеет огромное количество приложений. Вычисление числовой производной функции - одно из таких приложений. Следующая функция SML "d" вычисляет числовую производную заданной функции "f" в заданной точке "x":
- fun d delta fx = (f (x + delta) - f (x - delta)) / (2,0 * дельта); val d = fn: real ->(real ->real) ->real ->realЭта функция требует небольшого значения «дельта». Хорошим выбором для дельты при использовании этого алгоритма является кубический корень машины epsilon.
. Тип функции «d» указывает, что она отображает «float» на другую функцию с типом «(real ->реальный) ->реальный ->реальный ". Это позволяет нам частично применять аргументы. Этот функциональный стиль известен как каррирование. В этом случае полезно частично применить первый аргумент «дельта» к «d», чтобы получить более специализированную функцию:
- val d = d 1E ~ 8; val d = fn: (real ->real) ->real ->realОбратите внимание, что предполагаемый тип указывает, что замена «d» ожидает функцию с типом «real ->real» в качестве первого аргумента. Мы можем вычислить численное приближение к производной от at с:
- d (fn x =>x * x * x - x - 1.0) 3.0; val it = 25.9999996644: realПравильный ответ: ; .
Функция «d» называется «функцией высшего порядка», потому что она принимает другую функцию («f») как аргумент.
Каррированные функции и функции высшего порядка могут использоваться для устранения избыточного кода. Например, для библиотеки могут потребоваться функции типа
a ->b
, но удобнее писать функции типаa * c ->b
, где существует фиксированная связь между объекты типаa
иc
. Функция высшего порядка типа (a * c ->b) ->(a ->b) может исключить эту общность. Это пример шаблона адаптера .Дискретное вейвлет-преобразование (сопоставление с образцом)
1D вейвлет Хаара преобразование целого числа . Список чисел в степени в степени двух может быть очень сжато реализован в SML и является отличным примером использования сопоставления с образцом над списками, принимая пары элементов ("h1" и «div class="ht"») и сохраняя их суммы и разности в списках «s» и «d» соответственно:
- fun haar l = let fun aux [s] d = s :: d | вспомогательный s d = вспомогательный s d | aux (h1 :: div class="ht" :: t) s d = aux t (h1 + div class="ht" :: s) (h1-div class="ht" :: d) | aux _ _ _ = поднять пустой в конце вспомогательного l; val haar = fn: int list ->int listНапример:
- haar [1, 2, 3, 4, ~ 4, ~ 3, ~ 2, ~ 1]; val it = [0,20,4,4, ~ 1, ~ 1, ~ 1, ~ 1]: int listСопоставление с образцом - полезная конструкция, которая позволяет ясно и лаконично представлять сложные преобразования. Более того, компиляторы SML превращают сопоставление шаблонов в эффективный код, в результате чего программы становятся не только короче, но и быстрее.
РеализацииСуществует множество реализаций SML, в том числе:
- Standard ML of New Jersey (сокращенно SML / NJ) - это полный компилятор со связанными библиотеками, инструментами, интерактивным оболочка и документация. [1]
- Moscow ML - это облегченная реализация, основанная на среде исполнения CAML Light. Он реализует полный язык SML, включая модули SML и большую часть базовой библиотеки SML. [2]
- MLton - это компилятор оптимизации всей программы, который производит очень быстрый код по сравнению с другими реализациями ML. [3]
- В ML Kit интегрирован сборщик мусора (который можно отключить) и управление памятью на основе регионов с автоматическим определением регионов с целью поддержки в реальном времени Приложения. Его реализация очень близко основана на Определении.
- Poly / ML - это полная реализация Standard ML, которая производит быстрый код и поддерживает многоядерное оборудование (через потоки Posix); его система времени выполнения выполняет параллельную сборку мусора и онлайн-совместное использование неизменяемых подструктур.
- Isabelle / ML интегрирует параллельный Poly / ML в интерактивное средство доказательства теорем со сложной IDE (на основе jEdit ) для официальный стандарт ML (SML'97), диалект Isabelle / ML и язык доказательства. Начиная с Isabelle2016, существует также отладчик уровня исходного кода для ML.
- CakeML - версия ML цикла чтения-оценки-печати с формально проверенной средой выполнения и переводом на ассемблер
- HaMLet - это SML интерпретатор, который стремится быть точной и доступной эталонной реализацией стандарта.
- TILT - это полностью сертифицирующий компилятор для SML. Он использует типизированные промежуточные языки для оптимизации кода и обеспечения корректности и может компилироваться в типизированный язык ассемблера.
- SML.NET позволяет компилировать в Microsoft CLR и имеет расширения для связывания с другими .NET code.
- SML2c - это пакетный компилятор, который компилирует только объявления уровня модуля (т.е. подписи, структуры, функторы) в C. Он основан на SML / NJ версии 0.67 и использует интерфейсную часть и большую часть своей системы времени выполнения, но не поддерживает отладку и профилирование в стиле SML / NJ. Программы уровня модуля, которые работают на SML / NJ, могут быть скомпилированы с помощью sml2c без изменений.
- Система Poplog реализует версию SML с POP-11, и, возможно, Common Lisp и Prolog, позволяющие программировать на разных языках. Для всех языком реализации является POP-11, который компилируется постепенно. Он также имеет встроенный Emacs -подобный редактор, который взаимодействует с компилятором.
- SML # - это расширение SML, обеспечивающее полиморфизм записей и взаимодействие с языком C. Это обычный собственный компилятор, и его название не является намеком на работу на платформе.NET.
- Алиса : интерпретатор для Standard ML от Саарландского университета, добавляющий функции для ленивых вычислений, параллелизм (многопоточность и распределенные вычисления через вызовы удаленных процедур ) и программирование с ограничениями.
- SOSML является реализацией SML, написанного на TypeScript, который запускается непосредственно в веб-браузере. Он реализует большую часть языка SML и некоторые части базовой библиотеки SML.
Все эти реализации с открытым исходным кодом и находятся в свободном доступе. Большинство из них реализованы на SML. Коммерческих реализаций SML больше нет. Арлекин когда-то создал коммерческую IDE и назвал компилятор для SML. Компания сейчас не существует. перешел к Xanalys, а затем был приобретен 26 апреля 2013 года и с открытым исходным кодом.
Основные проекты с использованием SMLВся корпоративная архитектура ИТ-университета Копенгагена реализована примерно в 100 000 строк SML, включая записи о персонале, расчет заработной платы, администрирование курса и обратная связь, управление студенческими проектами и веб-интерфейсы самообслуживания.
Помощники по проверке HOL4 и Isabelle написаны на SML.
SML широко используется разработчиками компиляторов и микросхем, например, ARM.
См. ТакжеСсылкиВнешние ссылки
- Standard ML Family GitHub Project
- Стандартный язык ML Мадс Тофте, Scholarpedia, 4 (2): 7515. doi:10.4249/scholarpedia.7515
- What is SML?
- What is SML '97?
- successor ML (sML) is intended to provide a vehicle for the continued evolution of ML, using Standard ML as a starting point.
- Programming in Standard ML
- Programming in Standard ML '97: An On-line Tutorial
- Univ. of Chicago - SML tutorial (slides)
- CSE341: Programming Languages, Dan Grossman, University of Washington. Also on Coursera and YouTube