= Year Int
-- Int это целые числа
-- Месяц
data Month
= January
| February
| March
| April
| May
| June
| July
| August
| September
| October
| November | December
data Day = Day Int
-- Неделя
data Week
= Monday
| Tuesday
| Wednesday
| Thursday
| Friday
| Saturday
| Sunday
-- Время
data Time = Time Hour Minute Second
data Hour
= Hour
Int
-- Час
data Minute = Minute Int
-- Минута
data Second = Second Int
-- Секунда
Одной из основных целей разработчиков Haskell была ясность. Они стремились создать язык, предложе-
ния которого будут простыми и понятными, близкий к языку спецификаций.
С символом |мы уже познакомились, он указывает на альтернативы, объединение пишется через пробел.
Так, фраза
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 True23
компилятор предупредит нас об ошибке. Наверное, вы думаете, что приведенный пример надуман, ведь
кому захочется составлять время из логических значений? Но когда вы пишете программу, часто процесс
работы складывается так: вы думаете над одним, пишете другое, а также планируете вернуться к третьему.
И знание того, что есть надежный компилятор, который не пропустит глупых ошибок, освобождает руки, вы
можете не заботиться о таких пустяках, как правильное построение предложения.
Отметим, что такой подход с разделением вычисления на две стадии и проверкой типов называется
статической типизацией . Есть и другие языки, в них типы лишь подразумеваются и программа сразу начинает
Типы | 15
вычисляться, если есть какие-то несоответствия, об ошибке программисту сообщит вычислитель, причём
только тогда, когда вычисление дойдёт до ошибки. Такой подход называют динамической типизацией .
Типы требуют серьёзных размышлений на начальном этапе, этапе определения базовых терминов и спо-
собов их комбинирования. Не упускаем ли мы что-то важное из виду, или, может быть, типы имеют слишком
общий характер и допускают ненужные нам предложения? Приходится задумываться. Но если типы подо-
браны удачно, они сами начинают подсказывать, как строить программу.
1.3 Значения
Итак, мы определили типами базовые понятия и способы комбинирования. Обычно это небольшой набор
слов. Например в логических выражениях всего лишь два слова. Можем ли мы на что либо рассчитывать с
таким словарным запасом? Оказывается, что да. Здесь на помощь приходят синонимы. Сейчас у нас в активе
Читать дальше