Posted in Lisp
Practical Common Lisp, Chapter 7
Comment
MACROS : STANDARD CONTROL CONSTRUCS
All programmers should be used to the idea that the definition of a language can include a standard library of functionality that is implemented in terms of the "core" language functionality that could have benn implemented by any programmer on top of the language if it hand't been defined as part of the standard library.
다른 언어에서 특정 기능이 부족할때 새로운 클래스나 메소드를 정의함으로서 standard library 를 확장할 수 있지만, Lisp 은 이 방법 외에도 Macro 라는 조금 다른 방식을 제공한다.
각 매크로는s-exp
가 어떻게 lisp forms 으로 확장될지를 결정하는 자신만의 Syntax 를 정의할 수 있다. 이런 Macro 의 기능으로 인해서 Language 자체의 Syntax 가 확장될 수 있다. 대표적인 예가 WHEN
, DOLIST
, LOOP
와 같은 control constructs 와 DEFUN
, DEFPARAMETER
과 같은 definitinal forms 들이다.
WHEN and UNLESS
(if condition then-form [else-form])
은 then-form
과 else-form
이 single lisp form 이어야 한다는 제한을 가지고 있다.
그렇기 때문에 스팸 메일을 받았을때, 폴더에 저장하고, 데이터베이스에 업데이트 하려고 할 때 if
구문을 다음과 같이 작성하면, 제대로 작동하지 않는다.
(if (spam-p current-message)
(file-in-spam-folder current-message)
(update-spam-database current-message))
이 코드의 경우 update-spam-database
함수는 if-then
이 아니라 else-then
에 붙어있는 form 으로 인식되어 의도했던 바 대로 실행되지 않는다.
progn
을 이용하면, 여러 form 을 묶을 수 있기 때문에 다음과 같이 작성하면 제대로 동작한다.
(if (spam-p current-message)
(progn
(file-in-spam-folder current-message)
(update-spam-database current-message)))
그러나 매번 progn
을 써야한다는 사실이 귀찮고, if
와 progn
의 조합으로 해당 역할을 해낼 수 있다는걸 깨달았으므로, 다음과 같은 인터페이스를 가지는 when
매크로를 작성해보자.
(when (spam-p current-message)
(file-in-spam-folder current-message)
(update-spam-database current-message))
(defmacro when (condition &rest body)
`(if ,condition (progn ,@body)))
(defmacro unless (condition &rest body)
`(if (not ,condition) (progn ,@body)))
COND
if
같은 경우도, 중첩되면 보기가 싫어질 수 있다.
(if a
(do-x)
(if b
(do-y)
(do-z)))
이를 위해서 cond
매크로를 만들어 보자.
(defmacro cond (&rest body)
(when body
(let ((clause (first body)))
`(if ,(first clause) (progn ,@(rest clause))
(cond1 ,@(rest body))))))
AND, OR, and NOT
AND
와 OR
같은 경우 short-circuit 을 구현하기 위해서는, 두번 째 인자를 평가하지 않아야 하는데, 일반 함수로는 인자의 평가 시기를 조절하기 불가능하고, 매크로를 이용하면 그럴 수 있다.
;; AND and OR macros
(defmacro and1 (x &rest other)
`(if ,(not other)
,x
(if ,x (and1 ,@other) nil)))
(defmacro or1 (x &rest other)
`(if ,(not other)
,x
(if ,x ,x (or1 ,@other))))
Looping
Lisp 의 25가지 special operators 는 직접적으로 Looping 을 지원하지 않기 때문에 Lisp 의 모든 looping control constructs 는 이 specical operator 를 이용해 만든 Macro 다.
DO
는 강력하지만, less expressie 하기 때문에 DO
위에 만들어진 DOTIMES
와 DOLIST
를 제공한다. LOOP
는 non-lispy, english-like 스타일의 문법을 제공한다.
DOLIST and DOTIMES
아래는 DOLIST
와 DOTIMES
의 문법과 사용법이다.
;; syntax
(dolist (var list-form)
body-form*)
(dotimes (var count-form)
body-form*)
;; example
(dolist (x '(1 2 3)) (princ x) (princ x)) ;; 112233
(dotimes (x 4) (princ x) (princ x)) ;; 00112233
(dotimes (x 4) (print x) (if (= (/ x 2) 0) (return))) ;; 0 1 2
(dotimes (x 20)
(dotimes (y 20)
(format t "x: ~d, y: ~d~%" x y)))
DO
DO
의 Syntax 와 사용법은 아래와 같다.
;; Syntax
(do (variable-definitions*)
(end-test-form result-form*)
statement*)
;; Usage : fibonacci
(do ((n 0 (+1 n))
(cur 0 next
(next 1 (+ cur next)))
((=10 n) cur))
;; Usage: omiited result-form
;; same as (dotimes (i 4) (print i))
(do ((i 0 (1+ i)))
((>= i 4))
(print i))