EDIT: I changed the example code after the first answer because I came up with a simple version that begs the same questions.
I am currently learning Common Lisp's scoping properties. After I thought I had a solid understanding I decided to code up some examples that I could predict the outcome of, but apparently I was wrong. I have three question, each one relating to an example below:
Example 1:
(defmethod fun1 (x)
(print x)
(fun2))
(defmethod fun2 ()
(print x))
(fun1 5)
Output:
5
*** - EVAL: variable X has no value
Question: This makes sense. x is statically scoped and fun2 has no way of finding the value of x without having it passed explicitly.
Example 2:
(defvar x 100)
(defmethod fun1 (x)
(print x)
(fun2))
(defmethod fun2 ()
(print x))
(fun1 5)
Output:
5
5
Question: I don't understand why x is suddenly visible to fun2 with the value that fun1 gave it, instead of having a value of 100...
Example 3:
(setf x 100)
(defmethod fun1 (x)
(print x)
(fun2))
(defmethod fun2 ()
(print x))
(fun1 5)
Output:
5
100
Question: Should I ignore these results since calling setf on an undeclared variable is apparently undefined? This happens to be what I would expect in my second example...
Any insight would be greatly appreciated...
The effects of setting an undefined variable using
setf
is undefined in ANSI Common Lisp.defvar
will define a special variable. This declaration is global and also has effect onlet
bindings. That's the reason that by convention these variables are written as*foo*
. If you have ever definedx
withdefvar
, it is declared special and there is no way to declare it lexical later.let
by default provides local lexical variables. If the variable was already declared special (for example because of adefvar
), then it just creates a new local dynamic binding.Update
Nothing to see.
x
has been declared special. All uses of the variablex
now use dynamic binding. When calling the function, you bindx
to5
. Dynamically. Other functions can now access this dynamic binding and get that value.This is undefined behavior in Common Lisp. You are setting an undeclared variable. What happens then is implementation dependent. Your implementation (most do something similar) sets the symbol value of
x
to100
. Infun1
,x
is lexically bound. Infun2
evaluatingx
retrieves the symbol value (or possibly to the dynamically bound value) ofx
.As an example for an implementation that did (does?) something else: the CMUCL implementation would also have declare
x
in example 3 by default to be special. Setting an undefined variable also declares it special.NOTE
In portable standard compliant Common Lisp code the global variables are defined with
defvar
anddefparameter
. Both declare these variables to be special. ALL uses of these variables now involve dynamic binding.Remember:
is basically the same as
Which means that variable bindings in
let
bindings and variable bindings in function calls are working the same way. Ifx
would have been declared special in some place earlier, both would involve dynamic binding.This is specified in the Common Lisp standard. See for example the explanation to the SPECIAL declaration.