Визначення функцій в Ліспі
Змінимо наш підхід до побудови функції. В другому стовпчику побудовано функцію MEMBER, в основі якої лежить рекурсивний підхід, який базується на наступних фактах:
1. Якщо список порожній (не має елементів), то nam не належить списку.
2. Якщо nam дорівнює голові списку, то nam належить списку.
3. Якщо nam не дорівнює голові списку, то nam може належити списку тоді і тільки тоді коли nam належить хвосту списку.Розглянемо дві рекурсивні функції: REMBER (REMove memBER), яка має два аргументи — атом obj та список lst і яка видаляє перше зустрічання атома obj в списку lst. REMBER-ALL яка видаляє всі атоми obj в списку lst.
$ (DEFUN REMBER (obj lst)(DEFUN REMBER-ALL (obj lst)
((NULL lst) NIL)((NULL lst) NIL)
((EQL obj (CAR lst)) (CDR lst))((EQL obj (CAR lst))
(CONS (CAR lst)(REMBER-ALL obj (CDR lst))
(REMBER obj (CDR lst))) )(CONS (CAR lst)
(REMBER-ALL obj (CDR lst))) )
Результат роботи цих функцій проілюструємо на прикладах:
$ (REMBER ‘a ‘(q a w e r t a y))$ (REMBER-ALL ‘a ‘(q a w e r t a y))
(q w e r t a y)(q w e r t y)
Примітивна функція EQL використовується для порівняння атомів. Часто виникає потреба порівнювати списки. Напишемо функцію EQLIST, яка порівнює списки. Її побудуємо на основі наступних фактів:
1. Якщо перший список порожній, то, якщо і другий список порожній, повернути Т, інакше повернути NIL (або просто повернути (NULL другого списку)).
2. Якщо другий список порожній, повернути NIL.
3. Якщо голова першого списку не дорівнює голові другого списку, повернути NIL.
4. Перевірити рівність хвостів першого та другого списків.
$ (DEFUN EQLIST (lst1 lst2)$ (DEFUN NOT (obj)
((NULL lst1) (NULL lst2))(EQL obj NIL) )
((NULL lst2) NIL)
((NOT (EQL (CAR lst1) (CAR lst2))) NIL)
(EQLIST (CDR lst1) (CDR lst2)) )