在Common Lisp中有所谓的包(packages)的概念。包是符号的注册表,用作符号的命名空间。你可以请求Common Lisp列出所有的包。
CL-USER 1 > (list-all-packages)
(#<The SQL-COMMON package, 0/4 internal, 28/32 external>
#<The LOOP package, 245/256 internal, 3/4 external>
#<The COMM package, 0/4 internal, 940/1024 external>
#<The REG package, 41/64 internal, 0/4 external>
...)
每个包在某个数据结构中存储了内部符号。你可以询问Common Lisp哪些符号在一个包中被内化了。
CL-USER 2 > (loop for symbol being
each external-symbol in (find-package "COMMON-LISP")
collect symbol)
(MAKE-ARRAY INVOKE-DEBUGGER STRING-TRIM ...)
为了使这更加容易,Common Lisp提供了函数APROPOS和APROPOS-LIST。
CL-USER 3 > (apropos "MAKE-LOCK")
MP::INTERNAL-MAKE-LOCK (defined)
MP:MAKE-LOCK (defined)
WWW-UTILS:MAKE-LOCK (defined)
MAKE-LOCK
RESOURCES::MAKE-LOCK (defined)
MINIPROC:MAKE-LOCK (defined)
函数、类等使用符号作为它们的标识符。您也可以查询一个符号,它代表哪个函数。
CL-USER 4 > (symbol-function 'www-utils:make-lock)
#<Function WWW-UTILS:MAKE-LOCK 41E006A69C>
有时候,Common Lisp 也会记录函数的定义。这时可以使用 FUNCTION-LAMBDA-EXPRESSION 函数来检索它。
CL-USER 5 > (defun foo (a) (* (sin a) a))
FOO
CL-USER 6 > (pprint (function-lambda-expression 'foo))
(LAMBDA (A)
(DECLARE (SYSTEM::SOURCE-LEVEL #<EQ Hash Table{0} 41403151C3>))
(DECLARE (LAMBDA-NAME FOO))
(* (SIN A) A))
但是通常现代的Common Lisp实现并不会使用已记录的定义,而是记录每个Lisp构造的源位置。
大多数Common Lisp实现可以以特定于实现的方式跟踪源位置。
Common Lisp标准定义了一个名为ED的函数。
CL-USER 7 > (ed 'www-utils:make-lock)
这将调用一个编辑器(内部或外部),并应该打开该函数的源代码。为了使其正常工作,Common Lisp需要跟踪每个函数的源位置。接下来,编辑器需要访问该源代码。有时记录的位置是绝对路径/Users/joswig/lisp/utils.lisp。如果编辑器想要打开该文件,它应该是可访问的。但也可以使用逻辑路径名比如http:server;utils.lisp。然后将其转换为实际的物理路径名。这个转换稍后可以进行配置。因此,即使将Lisp移动到具有不同路径名的不同计算机上,配置逻辑路径名HTTP,Lisp仍然能够找到所有源代码,即使它在具有不同文件系统结构的不同计算机上。因此,为使其正常工作可能需要一些配置。但它是一个非常有用的功能,并且被广泛使用。
源代码的记录以及源位置的记录方式取决于实现并与其开发环境的功能相关。 更好的 Lisp实现在这个领域拥有很多特性。