Лямбда - выражения и пользовательские функции
Введём понятие формы. Формой в языке Lisp называется любая управляющая конструкция (т.е. конструкция, которая может быть вычислена). Примерами форм может быть (+ 1 2) или (LIST 'A '(B C) 'D). Надо отметить, что оператор, содержащий в себе формы, сам может являться формой. Атомы также являются формами.
Форма является важным понятием, с помощью которого мы в дальнейшем будем описывать вызовы функций. Если в предыдущей главе все функции работали с атомами или списками, то здесь будут рассматриваться функции, принимающие своими параметрами управляющие конструкции языка Lisp.
Основой любых вычислений в языке Lisp являются лямбда-выражения, которые, в свою очередь, берут за основу теорию лямбда-исчисления Черча (описанную в разделе 2).
В Lisp общее представление лямбда-выражения имеет вид (LAMBDA (X1 X2 … XN) Fn), где символ LAMBDA указывает на определение функции.
lambda | (LAMBDA ( (формальные_параметры) тело_функции) Создаёт функцию. Список формальных параметров (или так называемый лямбда-список) представляет собой перечисление используемых в теле функции имён переменных. Тело функции может являться произвольной формой. |
Для того чтобы понять, зачем это нужно, необходимо обратить внимание на форму записи вызовов функций (или лямбда-вызовов) в языке Lisp. Первый элемент списка всегда считается именем функции. Таким образом, можно в качестве некоторого первого элемента списка поставить лямбда-функцию и вычислить её от остальных элементов. С другой стороны, лямбда-вызов может быть также и фактическим параметром некоторой другой функции. Надо отметить, что если лямбда-выражение не используется одним из перечисленных способов, это приводит к ошибке.
В качестве примера приведём вычисления квадрата числа:
>(LAMBDA (X) (* X X)) #<Interpreted Function (unnamed) @ #x204d51ea> >((LAMBDA (X) (* X X)) 2) |
Как мы видим, использование ламбда-выражения без вызова привело к бессмысленному результату, однако правильно сформированный лямбда-вызов дал корректный результат.
В некотором приближении лямбда-выражение может считаться функцией без имени. Однако обычно гораздо удобнее формировать описание функции один раз, давать ему имя и впоследствии при вызове обращаться к функции по имени, нежели каждый раз подставлять лямбда-выражение, т.е. полное описание последовательности действий. Некоторые версии Lisp позволяют связывать лямбда-выражения с символами, и впоследствии использовать эти символы в качестве имён, однако данный метод не является общепринятым. Наиболее общим определением имени функции является использование DEFUN:
defun | (DEFUN имя_функции лямбда-смисок тело_функции) Определяет именованную функцию. Лямбда-список - список параметров, тело - произвольная форма. Впоследствии при обращении к имени функции на его место подставляется лямбда-выражение, составленное из лямбда-списка и тела функции. |
Возвращаясь к вычислению квадрата числа, мы можем привести аналогичный код, но уже с использованием DEFUN:
>(DEFUN SQUARE (X) (* X X)) SQUARE >(SQUARE 2) |
В дальнейшем каждое обращение к SQUARE (до тех пор, пока символ не будет переопределён) будет вычислять квадрат аргумента.
Можно проверить, связан ли символ с функцией. Для этой цели используется функция FBOUNDP:
fboundp | (FBOUNDP символ) Определяет, связано ли с символом определение функции и возвращает истину если связано и ложь в противном случае. |
>(FBOUNDP 'SQUARE) T >(FBOUNDP 'SQUARE1) NIL |
Данный пример был выполнен в Сlisp. В случае Allegro Common Lisp результат будет другой:
>(FBOUNDP 'SQUARE) #<Interpreted Function SQUARE> |
Тем не менее, будучи использован в аргументах других функций, этот вывод на экран соответствует возвращаемому значению T.
Ещё одна любопытная возможность связана со способностью Lisp показывать, каким именно образом функция была определена.
Symbol-function | (symbol-function символ) Возвращает текстовое определение функции. |
>(SYMBOL-FUNCTION 'SQUARE) #<CLOSURE SQUARE (X) (DECLARE (SYSTEM::IN-DEFUN SQUARE)) (BLOCK SQUARE (* X X))> |
Данный пример также выполнен в Clisp.
Для списка аргументов функций допустимо также указывать дополнительные свойства, которые начинаются со знака & и записываются перед объявляемым аргументом функции (использование таких слов – отличительная особенность диалекта COMMON LISP). Это свойства &OPTIONAL, &REST, &KEY и &AUX. Все параметры, объявленные до первого &KEY являются обязательными. Далее, значения параметров, объявленных через &OPTION можно не указывать, и они будут переданы в тело функции как NIL или значением по умолчанию (если оно указано).
>(defun sample (x &optional (y 5)) (* x y)) SAMPLE > (sample 1) >(sample 1 2) >(defun sample2 (&optional y) y) SAMPLE2 >(sample2 1) >(sample2) NIL |
Все параметры, которые указываются после слова &REST, передаются в тело программы через список несвязанных параметров (для организации функций с переменным числом параметров).
>(defun sample3 (&rest params) params) SAMPLE3 >(sample3) NIL >(sample3 'A) (A) >(sample3 'A 'B 'C) (A B C) |
Отметим, что наряду с (CONS x NIL) подобная функция также может использоваться для задачи помещения атома в скобки.
Параметры, объявленные со свойством &KEY являются необязательными и могут быть переданы в функцию только в виде пары :ИМЯ ЗНАЧЕНИЕ, зато в любом порядке:
>(defun sample4 (&key x y) (list x y)) SAMPLE4 >(sample4) (NIL NIL) >(sample4 :x 1) (1 NIL) >(sample4 :y 1) (NIL 1) |
>(sample4 :x 1 :y 2) (1 2) |
Наконец, параметры, объявленные как вспомогательные, являются скорее локальными переменными, нежели аргументами функции и не требуют передачи при вызове.