С символом | мы уже познакомились, он указывает на альтернативы, объединение пишется через пробел.
Так, фраза
data Time = Time Hour Minute Second
означает, что тип Time – это значение с меткой Time, которое состоит из значений типов “час”, “время” и
“секунда”, и больше ничего. Метку принято называть конструктором.
Фраза
data Year = Year Int
означает, что тип Year – это значение с конструктором Year, которое состоит из одного значения типа
Int. Конструктор обычно идёт первым, а за ним через пробел следуют другие типы. Конструктор может быть
и самостоятельным значением, как в случае True или January.
Типы делят выполнение программы на две стадии: компиляцию (compile-time) и вычисление (run-time). На
этапе компиляции происходит проверка типов. Программа, которая не прошла проверку типов, считается
бессмысленной и не вычисляется. Приложение, которое выполняет компиляцию, называют компилятором
(compiler), а то приложение, которое проводит вычисление, называют вычислителем (run-time system).
Типами мы определяем основные понятия в том явлении, которое мы хотим описать, а также осмыслен-
ные способы их комбинирования. Мы говорим, как из простейших терминов получаются составные. Если мы
попытаемся построить бессмысленное предложение, компилятор языка автоматически найдёт такое предло-
жение и сообщит нам об этом. Этот процесс заключается в проверке типов, к примеру если у нас есть функция
сложения чисел, и мы попытаемся передать в неё строку или список, компилятор заметит это и скажет нам
об этом перед тем как программа начнёт выполнятся. И важно то, что это произойдёт очень быстро. Если мы
случайно ошиблись в выражении, которое будет вычислено через час, нам не нужно ждать пока вычислитель
дойдёт до ошибки, мы узнаем об этом, не успев моргнуть, после запуска программы.
Итак, если мы попробуем составить время из месяцев и логических значений:
Time January True 23
компилятор предупредит нас об ошибке. Наверное, вы думаете, что приведенный пример надуман, ведь
кому захочется составлять время из логических значений? Но когда вы пишете программу, часто процесс
работы складывается так: вы думаете над одним, пишете другое, а также планируете вернуться к третьему.
И знание того, что есть надежный компилятор, который не пропустит глупых ошибок, освобождает руки, вы
можете не заботиться о таких пустяках, как правильное построение предложения.
Отметим, что такой подход с разделением вычисления на две стадии и проверкой типов называется
статической типизацией. Есть и другие языки, в них типы лишь подразумеваются и программа сразу начинает
Типы | 15
вычисляться, если есть какие-то несоответствия, об ошибке программисту сообщит вычислитель, причём
только тогда, когда вычисление дойдёт до ошибки. Такой подход называют динамической типизацией.
Типы требуют серьёзных размышлений на начальном этапе, этапе определения базовых терминов и спо-
собов их комбинирования. Не упускаем ли мы что-то важное из виду, или, может быть, типы имеют слишком
общий характер и допускают ненужные нам предложения? Приходится задумываться. Но если типы подо-
браны удачно, они сами начинают подсказывать, как строить программу.
1.3 Значения
Итак, мы определили типами базовые понятия и способы комбинирования. Обычно это небольшой набор
слов. Например в логических выражениях всего лишь два слова. Можем ли мы на что либо рассчитывать с
таким словарным запасом? Оказывается, что да. Здесь на помощь приходят синонимы. Сейчас у нас в активе
лишь два слова:
data Bool = True | False
И мы можем определить два синонима:
true :: Bool
true = True
false :: Bool
false = False
В Haskell синонимы пишутся с маленькой буквы. Синоним определяется через знак =. Обратите внимание
на то, что это не процесс вычисления значения. Мы всего лишь объявляем новое имя для комбинации слов.
Теперь мы имеем целых четыре слова! Тем не менее, ушли мы не далеко, и два новых слова, в сущности,
не делают язык выразительнее. Такие синонимы называют константами. Это значит, что одним словом мы
будем обозначать некоторую комбинацию других слов. В данном случае комбинации очень простые.
Но наши синонимы могут определять одни слова через другие. Синонимы могут принимать параметры.