所以,我已经阅读
了 lisp 中的 setq 和 defvar,
http://www.cs.ucf.edu/courses/cop4020/spr2006/plsetup.html和
在 Lisp 中,我该如何修复“警告:假定特殊?”
在其他地方关于 setf 和 defvar 之间的区别。所以我决定尝试一下这个想法:
CL-USER> (defun foo ()
(setf x 10)
(print x))
; in: DEFUN FOO
; (SETF X 10)
; ==>
; (SETQ X 10)
;
; caught WARNING:
; undefined variable: X
;
; compilation unit finished
; Undefined variable:
; X
; caught 1 WARNING condition
FOO
CL-USER> x
; Evaluation aborted on #<UNBOUND-VARIABLE X {10040F1543}>.
CL-USER> (foo)
10
10
CL-USER> x
10
好的,我知道应该使用 setf 来更改现有变量的值,但是未定义变量警告似乎在 SBCL 中处理得很好(尽管我已经读过不同的 CL 实现可能会以不同的方式处理这个问题,因此它不是最好的办法)。
进入第二个测试:
CL-USER> (defun bar ()
(defvar y 15)
(print y))
; in: DEFUN BAR
; (PRINT Y)
;
; caught WARNING:
; undefined variable: Y
;
; compilation unit finished
; Undefined variable:
; Y
; caught 1 WARNING condition
BAR
CL-USER> y
; Evaluation aborted on #<UNBOUND-VARIABLE Y {10045033D3}>.
CL-USER> (bar)
15
15
CL-USER> y
15
根据链接,我将 setf 更改为 defvar 我认为应该一次创建和绑定变量。现在我的未定义变量警告被推送到 (print y) 行......这里发生了什么?
作为第二个问题,我希望函数内分配的任何变量的值在函数之外都无法访问,就像 Python 中的情况一样:
>>> def foo():
... x = 10
... print x
...
>>> foo()
10
>>> x
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'x' is not defined
我猜这与 common lisp 处理范围的方式有关,即 defvar 创建了一个“全局特殊变量”......所以我最后一次尝试了 (let ...)
CL-USER> (defun baz () (let ((z 10)) (print z)) (incf z 10) (print z))
; in: DEFUN BAZ
; (INCF Z 10)
; --> LET*
; ==>
; (SETQ Z #:NEW0)
;
; caught WARNING:
; undefined variable: Z
;
; compilation unit finished
; Undefined variable:
; Z
; caught 1 WARNING condition
在阅读了 defvar、defparameter、setf 和 setq 之间的区别之后,这个似乎工作正常:
CL-USER> (defun apple ()
(defparameter x 10)
(print 10))
APPLE
CL-USER> x
; Evaluation aborted on #<UNBOUND-VARIABLE X {1004436993}>.
CL-USER> (apple)
10
10
CL-USER> x
10
重申我的问题:1) setf、defvar 和 let 到底发生了什么?
2) 有没有办法让 common lisp 像 python 示例中那样在函数内限定变量?