Обучение Common Lisp (с использованием GNU CLISP 2.43).. так что может быть ошибкой noob. Например, "печатные простые числа между x и y"
(defun is-prime (n)
(if (< n 2) (return-from is-prime NIL))
(do ((i 2 (1+ i)))
((= i n) T)
(if (= (mod n i) 0)
(return NIL))))
(defun next-prime-after (n)
(do ((i (1+ n) (1+ i)))
((is-prime i) i)))
(defmacro do-primes-v2 ((var start end) &body body)
`(do ((,var (if (is-prime ,start)
,start
(next-prime-after ,start))
(next-prime-after ,var)))
((> ,var ,end))
,@body))
(defmacro do-primes-v3 ((var start end) &body body)
(let ((loop-start (gensym))
(loop-end (gensym)))
`(do ((,loop-start ,start)
(,loop-end ,end)
(,var (if (is-prime ,loop-start)
,loop-start
(next-prime-after ,loop-start))
(next-prime-after ,var)))
((> ,var ,loop-end))
,@body )))
do-primes-v2 отлично работает.
[13]> (do-primes-v2 (p 10 25) (format t "~d " p)) 11 13 17 19 23
Далее я попытался использовать gensym, чтобы избежать именования столкновений при расширении макросов - do-primes-v3. Однако я застрял с
*** - EVAL: variable #:G3498 has no value
Пробовал использовать макро-расширение, чтобы увидеть, могу ли я обнаружить ошибку, но я не могу.
[16]> (macroexpand-1 `(do-primes-v3 (p 10 25) (format t "~d " p))) (DO ((#:G3502 10) (#:G3503 25) (P (IF (IS-PRIME #:G3502) #:G3502 (NEXT-PRIME-AFTER #:G3502)) (NEXT-PRIME-AFTER P))) ((> P #:G3503)) (FORMAT T "~d " P)) ;