Функциональный аргумент можно пометить с помощью специальной, предотвращающей вычисления формы FUNCTION:
(FUNCTION функция)
В отличие от обычной блокировки вычислений с помощью QUOTE форму FUNCTION называют функциональной блокировкой. Функциональную блокировку можно записывать сокращенно:
#’f Û (FUNCTION f)
Если нужно передать функции данные в том виде, как они записаны, то используется обычная форма quote. Функции quote достаточно и для передачи имени функции или лямбда-выражения, если в нем не используются свободные переменные. Работа со свободными переменными оказывается более сложной, чем с параметрами, поскольку значения свободных переменных зависят от контекста вычислений. Сформированный на время вычисления функции вычислительный контекст после окончания ее вычисления пропадает, и на него невозможно позже сослаться или вернуться к нему. Для запоминания контекста вычисления в Лиспе используется замыкание (лексическое замыкание) – пара, состоящая из функции (лямбда-выражения) и контекста.
Замыкание создается формой FUNCTION. В замыкание из контекста определения функции включаются лишь связи свободных переменных функции. Замыкание можно сохранять как любой лисповский объект, присваивая его какой-нибудь переменной:
>(defun add(x)
(function (lambda (y) (+ x y))))
ADD
>(setq add3 (add 3)); x = 3
#<Closure SPECIAL::APPLY-INTERPRETED-CLOSURE...>
>(funcall add3 5); y = 5
Связи свободных переменных замыкания остаются в силе до следующего запуска замыкания, и этим переменным можно даже присваивать новые значения.
Необходимость замыканий продемонстрируем на следующем примере. Допустим, Вы хотите написать программу, которая при вызове генерирует степени двойки. Это должно выглядеть следующим образом:
>(power-of-2-generator)
>(power-of-2-generator)
>(power-of-2-generator)
...
Один из способов написать такой генератор – использовать глобальные переменные:
(defvar *previous-power* 1)
(defun power-of-2-generator ()
(setf *previous-power* (* 2 *previous-power*)))
Функция работает, но она использует глобальные переменные, а это значит, что мы не можем иметь больше одного power-of-2-generator генератора одновременно, так как глобальная переменная *previous power* может сохранять только одно значение.
Можно ли обойтись без глобальных переменных? Для сохранения состояния генератора может быть использовано замыкание. Связи свободных переменных замыкания остаются в силе до следующего запуска замыкания. Этим переменным также можно присваивать значения. Используя замыкание, получим следующее определение генератора степеней двойки:
(defun make-power-of-2-generator (previous-power)
#'(lambda ()
(setf previous-power (* previous-power 2))))
Теперь мы можем создать несколько генераторов, так как каждый из них использует собственную переменную previous-power.
>(setf g1 (make-power-of-2-generator 1))
#<Closure SPECIAL::APPLY-INTERPRETED-CLOSURE...>
>(setf g2 (make-power-of-2-generator 10))
#<Closure SPECIAL::APPLY-INTERPRETED-CLOSURE...>
>(funcall g1)
>(funcall g2)
>(funcall g1)
>(funcall g2)
>(funcall g1)