Common Lisp 有包的概念。包是一种符号注册表,用作符号的命名空间。您可以向 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 结合其开发环境的一个特性。更好的 Lisp 实现在这方面有很多特性。