Common Lisp 超规范(译文):11.包

11. 包

11.1 包概念

11.1.1 包的介绍

一个包[package]建立一个从名字到符号[symbol]的映射. 在给定的任何时间, 都有一个包[package]是当前的. 当前包[current package]是 *package* 的值[value]的那个包. 当使用 Lisp 读取器[Lisp reader]时, 可以去引用非当前包[package]中的符号[symbol], 通过在那个符号[symbol]的打印表示中使用包前缀[package prefixe].

下面这段列出了一些可应用于包[package]的已定义的名字[defined name]. 在一个操作符[operator]接收一个符号[symbol]或一个符号[symbol]列表[list]作为参数的地方, 一个 nil 参数被当作是一个空的符号[symbol]列表[list]. 任何包 package 参数可能是一个字符串[string], 一个符号[symbol], 或者一个包[package]. 如果提供了一个符号[symbol], 它的名字会被用作这个包[package]的名字.

*modules*            import                     provide           
*package*            in-package                 rename-package    
defpackage           intern                     require           
do-all-symbols       list-all-packages          shadow            
do-external-symbols  make-package               shadowing-import  
do-symbols           package-name               unexport          
export               package-nicknames          unintern          
find-all-symbols     package-shadowing-symbols  unuse-package     
find-package         package-use-list           use-package       
find-symbol          package-used-by-list                         

Figure 11-1. 一些和包相关的已定义的名字
11.1.1.1 包名和别名

每个包[package]都有一个名字[name] (一个字符串[string]) 并且或许有一些别名[nickname] (也是字符串[string]). 在这个包被创建时它们被赋值并且可以在后面更改.

这里有一个包[package]的单独的命名空间. 函数[function] find-package 把一个包的名字[name]或别名[nickname]转换为关联的包[package]. 函数[function] package-name 返回一个包[package]的名字[name]. 函数[function] package-nicknames 返回一个包[package]的所有别名[nickname]的列表[list]. rename-package 移除一个包[package]的当前名字[name]和别名[nickname]并且用调用者指定的新的那些替换它们.

11.1.1.2 一个包中的符号
11.1.1.2.1 内部和外部符号

在一个包[package]中的映射被分为两类, 外部的和内部的. 这些不同的映射所针对的符号[symbol]被称为这个包[package]的外部符号[external symbol]和内部符号[internal symbol]. 在一个包[package]中, 一个名字引用一个符号[symbol]或者没有引用; 如果它确实引用一个符号[symbol], 那么它可能是那个包中的外部符号或内部符号, 但是不可能都是. 外部符号[external symbol]是这个包对于其他包[package]的公共接口部分. 如果符号[symbol]从一个给定包[package]中被导出[export], 那么它们就是这个包[package]的外部符号[external symbol].

一个符号[symbol]不管出现[present]在哪个包[package]中它都有着相同的名字, 但是它可能是某些包[package]的一个外部符号[external symbol], 其他包的内部符号[internal symbol].

11.1.1.2.2 包的继承

包[package]可以分层构建. 从某个角度, 一个包[package]是一个从字符串[string]到内部符号[internal symbol]和外部符号[external symbol]的映射的单独集合. 然而, 这些映射中的一部分可能在这个包[package]自身中被建立, 而其他映射可能通过 use-package 从其他包[package]中继承而来. 如果一个映射是在那个包[package]自身中的并且不是从某个其他地方继承来的, 就说一个符号[symbol]出现[present]在一个包[package]中.

这里没有方法去继承另一个包[package]的内部符号[internal symbol]; 在使用 Lisp 读取器[Lisp reader]时, 为了引用一个内部符号[internal symbol], 包含这个符号[symbol]的包[package]一定是当前包[current package], 或者一定使用一个包前缀[package prefix], 或者这个符号[symbol]一定被导入到当前包[current package]中.

11.1.1.2.3 一个包中符号的可访问性

如果一个包[package]在一个符号[symbol]被创建时是它的 home 包[home package], 或者它被导入到这个包[package]中, 或者通过 use-package 继承, 那么这个符号[symbol]在这个包[package]中就是可访问的[accessible].

如果一个符号[symbol]在一个包[package]中是可访问的[accessible], 当那个包[package]是当前包[current package]时用 Lisp 读取器, 它可以在不带包前缀[package prefix]的情况下被引用, 不管它是出现[present]在那个包中还是继承的.

来自一个包[package]中的符号[symbol]可以用两种方式使得它在另一个包[package]中可访问[accessible].

– 任何单独符号[symbol]可以通过使用 import 被添加到一个包[package]中. 在这个对 import 的调用后那个符号[symbol]就出现[present]在那个导入的包[package]中. 在这个符号[symbol]来源的包[package]中(如果有的话)的该符号[symbol]的状态是不会被改变, 并且这个符号[symbol]的 home 包[home package]没有被改变. 一旦被导入, 一个符号[symbol]就出现[present]在那个导入的包[package]中并且只能通过调用 unintern 来移除.

一个符号[symbol]被某个包中的另一个符号[symbol]所遮蔽[shadow[3]], 如果第二个符号没有出现第一个符号[symbol]通过继承是可访问[accessible]的话. 见 shadowing-import.

– 使一个包[package]中的符号[symbol]在另一个包中是可访问[accessible]的第二种机制是由 use-package 提供的. 那个被使用的包[package]中的所有外部符号[external package]都被那个使用的包[package]所继承. 函数[function] unuse-package 撤销一个之前的 use-package 的影响.

11.1.1.2.4 查找一个包中的一个符号

当要在一个给定的包中查找一个符号时会发生以下情况:

– 对这个包[package]的外部符号[external package]和内部符号[internal symbol]进行搜索来查找这个符号[symbol].
– 所使用的包[package]的外部符号[external package]会以某个未指定的顺序进行搜索. 顺序无关紧要; 见下面列出的用于处理名字冲突的规则.

11.1.1.2.5 包中的名字冲突的避免

在一个包[package]中, 任何独有的名字最多只能引用一个符号[symbol]. 当这里有超过一个候选符号[symbol]时就说发生了名字冲突. 任何一个名字冲突将要发生时, 就会发出一个可校正的[correctable]错误[error].

以下规则应用于名字冲突:

– 当名字冲突成为可能时它们会被检测到, 就是说, 在这个包结构被修改时. 在每次名字查找时名字冲突不会被检测.

– 如果对于一个包[package]相同[same]符号[symbol]可以通过超过一种途径访问, 那么这里没有名字冲突. 一个符号[symbol]不能和自身冲突. 名字冲突只发生在有着相同名字(在 string= 下)的不同[distinct]符号[symbol]之间.

– 每个包[package]都有一个遮蔽符号[symbol]的列表. 一个遮蔽符号[symbol]优先于其他在那个包[package]中可访问的[accessible]任何相同名字的符号[symbol]. 一个涉及到遮蔽符号的名字冲突总是在没有发出一个错误的情况下, 以选择遮蔽符号的形式得到解决 (除了一个涉及 import 的例外情况). 见 shadow 和 shadowing-import.

– 函数 use-package, import, 和 export 检测名字冲突.

– shadow 和 shadowing-import 从不发出一个名字冲突的错误.

– unuse-package 和 unexport 不需要去做任何名字冲突的检测. unintern 只有在一个被解除捕捉[uninterned]的符号[symbol]是一个遮蔽符号[shadowing symbol]时执行名字冲突检测.

– 给 unintern 传递一个遮蔽符号可以发现一个之前通过遮蔽解决的名字冲突.

– 包函数在对包结构做任何改动之前发出一个 package-error 类型[type]的名称冲突错误. 当需要进行多次更改时, 允许具体实现分别处理每一个更改. 比如, 当 export 给定一个符号[symbol]的列表[list]时, 从列表[list]中第二个符号[symbol]导致的名字冲突终止可能仍然会导出这个列表[list]的第一个符号[symbol]. 然而, 一个由单个符号的 export 导致的名字冲突错误会在在任何包[package]中那个符号[symbol]的可访问性[accessibility]被改变之前被发出.

– 从一个名字冲突错误中的继续必须为用户提供一个机会去选择任意一个候选符号来解决名字冲突. 包[package]结构应该通过 shadowing-import, unintern, 或 unexport 被修改, 以反映名字冲突的解决方案.

– 这个 use-package 中, 出现[present]在使用的包[package]中的符号[symbol]和被使用包中的外部符号[external symbol]之间的名字冲突, 通过使第一个符号[symbol]变为遮蔽符号[symbol]进而选择第一个符号或者通过从使用的包[package]中解绑第一个符号[symbol]进而选择第二个符号[symbol]来解决.

– 在 export 或 unintern 中由于一个包[package]从两个其他包[package]中继承相同[same]名字[name] (在 string= 下)的不同[dictinct]符号[symbol]而导致的一个名字冲突, 可以通过把任意一个符号[symbol]导入到使用的包[package]中并且使它成为一个遮蔽符号[shadowing symbol]来解决, 就像使用 use-package 一样.

11.1.2 标准包

这个章节描述了在每个符合规范的实现[conforming implementation]中都是可用的包[package]. 下面这段中提供了这些标准[standardized]包[package]的名字[name]和别名[nickname]的汇总.

名称              别名  
COMMON-LISP       CL         
COMMON-LISP-USER  CL-USER    
KEYWORD           none       

Figure 11-2. 标准包名称
11.1.2.1 COMMON-LISP 包

这个 COMMON-LISP 包包含了这个规范定义的 Common Lisp 系统的基础. 它的外部符号[external symbol]包含了出现在 Common Lisp 系统中的所有已定义的名字[defined name] (除了 KEYWORD 包中已定义的名字[defined name]外), 例如 car, cdr, *package*, 等等. 这个 COMMON-LISP 有着别名[nickname] CL.

这个 COMMON-LISP 包有着章节 1.9 (COMMON-LISP 包中的符号) 中枚举的那些外部符号[external symbol], 没有其他的. 这些外部符号[external symbol]出现[present]在 COMMON-LISP 包中但是它们的 home 包[home package]不需要是 COMMON-LISP 包.

比如, 符号 HELP 不能是 COMMON-LISP 包的一个外部符号[external symbol]因为它没有在章节 1.9 (COMMON-LISP 包中的符号) 中被提及. 与此相反, 符号[symbol] variable 必须是 COMMON-LISP 包的一个外部符号[external symbol]即便它没有定义, 因为它在那个章节中被列出来了 (为了支持它作为给函数[function] documentation 的第二个有效实参[argument]的使用).

这个 COMMON-LISP 包有着额外的内部符号[internal symbol].

11.1.2.1.1 符合规范的具体实现的 COMMON-LISP 包的约束

在一个符合规范的具体实现[conforming implementation]中, 这个 COMMON-LISP 包的一个外部符号[external symbol]可以有一个函数[function], 宏[macro], 或者特殊操作符[special operator]定义, 一个全局变量[global variable]定义 (或者由于一个 special 公告[proclamation]作为一个动态变量[dynamic variable]的其他状态), 或者只有在这个标准中显式允许时可以是一个类型[type]定义. 比如, fboundp 对于 COMMON-LISP 包的任何不是一个标准[standardized]函数[function], 宏[macro]或特殊操作符[special operator]的名字[name]的外部符号[external symbol]产生[yield] false, 而 boundp 对于 COMMON-LISP 包中的任何不是标准[standardized]全局变量[global variable]名字的外部符号[external symbol]返回 false. 此外, 符合规范的程序[conforming program]可以使用 COMMON-LISP 包的外部符号[external symbol]作为局部词法变量[lexical variable]的名字[name], 并相信这些名字[name]没有被具体实现全局声明为 special, 除非这些符号[symbol]是标准[standardized]全局变量[global variable]的名字[name].

一个符合规范的具体实现[conforming implementation]一定不能用任何标准[standardized]包[package]的外部符号[external symbol]或者 COMMON-LISP-USER 包中可访问[accessible]符号[symbol]作为属性指示符[property indicator]在 COMMON-LISP 包的外部符号[external symbol]上放置任何属性[property].

11.1.2.1.2 符合规范的程序的 COMMON-LISP 包的约束

除了显式允许的地方, 如果在一个 COMMON-LISP 包的外部符号[external symbol]上执行以下任何动作, 后果都是未定义的:

  1. 绑定[binding]或修改它的值 (词法地或动态地). (下面记录一些异常情况.)

  2. 定义, 解除定义, 或者绑定[binding]它为一个函数[function]. (下面记录一些异常情况.)

  3. 定义, 解除定义, 或绑定[binding]它为一个宏[macro]或编译器宏[compiler macro]. (下面记录一些异常情况.)

  4. 定义它为一个类型指定符[type specifier] (通过 defstruct, defclass, deftype, define-condition).

  5. 定义它为一个结构体 (通过 defstruct).

  6. 使用 declaration 全局声明[proclamation]定义它为一个声明[declaration].

  7. 定义它为一个符号宏[symbol macro].

  8. 修改它的 home 包[home package].

  9. 追踪它 (通过 trace).

  10. 声明或全局声明它为 special (通过 declare, declaim, 或 proclaim).

  11. 声明或全局声明它的 type 或 ftype (通过 declare, declaim, or proclaim). (下面记录一些异常情况.)

  12. 从 COMMON-LISP 包中移除它.

  13. 为它定义一个 setf 展开器[setf expander] (提供 defsetf 或 define-setf-method).

  14. 定义, 解除定义, 或绑定它的 setf 函数名[setf function name].

  15. 定义它为一个方法组合类型[method combination] (通过 define-method-combination).

  16. 使用它作为给 find-class 的 setf 的 class-name 参数.

  17. 绑定它为一个捕捉标记[catch tag].

  18. 绑定它为一个重启动[restart]名字[name].

  19. 为标准[standardized]广义函数[generic function]定义一个当所有实参[argument]都是标准化[standardized]类[class]的直接实例[direct instance]时可应用[applicable]的方法[method].

11.1.2.1.2.1 符合规范的程序的 COMMON-LISP 包的约束的一些异常

如果 COMMON-LISP 包的一个外部符号[external symbol]没有被全局定义为一个标准[standardized]动态变量[dynamic variable]或常变量[constant variable], 那么允许词法上绑定[bind]它并且声明那个绑定[binding]的 type, 并且允许去局部地把它建立[establish]为一个符号宏[symbol macro] (比如, 使用 symbol-macrolet).

除非显式指定, 否则如果一个 COMMON-LISP 包的外部符号[external symbol]被全局定义为一个标准[standardized]动态变量[dynamic variable], 允许去绑定[bind]或赋值[assign]那个动态变量[dynamic variable], 前提是保持这个动态变量的 “值类型(Value Type)” 约束, 并且那个变量[variable]的新值[value]和该变量[variable]声明的目的一致.

如果一个 COMMON-LISP 包的外部符号[external symbol]没有被定义为一个标准[standardized]函数[function], 宏[macro], 或特殊操作符[special operator], 允许词法上把它绑定[bind]为一个函数[function] (比如, 使用 flet), 去声明那个绑定[binding]的 ftype, 并且去追踪(trace)那个绑定[binding] (在提供了执行这个的能力的具体实现中) .

如果一个 COMMON-LISP 包的外部符号[external symbol]没有被定义为一个标准[standardized]函数[function], 宏[macro], 或特殊操作符[special operator], 允许去词法上把它绑定[bind]为一个宏[macro] (比如, 使用 macrolet).

如果一个 COMMON-LISP 包的外部符号[external symbol]没有被定义为一个标准[standardized]函数[function], 宏[macro], 或特殊操作符[special operator], 允许去词法上把它的 setf 函数名[setf function name]绑定[bind]为一个函数[function], 并且去声明那个绑定[binding]的 ftype.

11.1.2.2 COMMON-LISP-USER 包

在 Common Lisp 系统启动时, 这个 COMMON-LISP-USER 包是当前包[current package]. 这个包[package]使用[use]了 COMMON-LISP 包. 这个 COMMON-LISP-USER 包有着别名[nickname] CL-USER. 这个 COMMON-LISP-USER 包可以有捕捉在它内部的额外符号; 它可以使用[use]其他具体实现定义[implementation-defined]的包[package].

11.1.2.3 KEYWORD 包

这个 KEYWORD 包包含了称之为关键字[keyword[1]]的符号[symbol], 通常被用作程序[program]以及它们关联的数据表达式[expression[1]]中的特殊标记.

以一个包标记[package marker]开始的符号[symbol]标记[token]被 Lisp 读取器[Lisp reader]解析为 KEYWORD 包中的符号[symbol]; 见章节 2.3.4 (符号标记). 这使得在不同包[package]中的程序之间进行通信时使用关键字[keyword]变得方便. 比如, 在调用[call]中传递关键字参数[keyword parameter]的机制使用关键字[keywords[1]]来命名相应的实参[argument]; 见章节 3.4.1 (普通 lambda 列表).

在 KEYWORD 包中的符号[symbol]按照定义是 keyword 类型[type]的.

11.1.2.3.1 在 KEYWORD 包中捕捉一个符号

KEYWORD 包的处理方式与其他包[package]不同, 因为一个符号[symbol]被捕捉[interned]到其中时, 会采取特殊的操作. 特别地, 当一个符号[symbol]被捕捉[interned]到 KEYWORD 包中时, 它自动成为一个外部符号[external symbol]并且自动成为一个以它自身作为值[value]的常变量[constant variable].

11.1.2.3.2 KEYWORD 包的注意点

一般情况下, 最好将关键字的使用限制在存在要被选择的有限可枚举的名称集合的情况下. 比如, 如果一个亮度切换有两个状态, 它们可能被称作 :on 和 :off.

在这个名称集合不是有限可枚举的情况下 (也就是说, 可能发生名字冲突的情况下) 最好去使用某个不是 KEYWORD 的包中的符号[symbol], 这样冲突自然会被避免. 比如, 一个程序[program]使用一个关键字[keyword[1]]作为属性指示符[property indicator]通常是不明智的, 因为如果这里有另一个程序[program]做了相同的事, 每一个都会把另一个的数据重写掉.

11.1.2.4 依赖于具体实现的包

其他的, 具体实现定义[implementation-defined]的包[package]可能出现在最初的 Common Lisp 环境中.

一个符合规范的具体实现[conforming implementation]的文档建议但不是必须去包含一个最初出现在那个具体实现[implementation]但没有在这个规范中指定的所有包名的列表. (也见函数[function] list-all-packages.)

11.2 包字典

系统类 PACKAGE

  • 类优先级列表(Class Precedence List):

      package, t
    
  • 描述(Description):

      一个包[package]是一个映射符号[symbol]名称[name]到符号[symbol]的命名空间[namespace]; 见章节 11.1 (包概念).
    
  • 参见(See Also):

      章节 11.1 (包概念), 章节 22.1.3.13 (打印其他对象), 章节 2.3.4 (符号标记) 
    

函数 EXPORT

  • 语法(Syntax):

      export symbols &optional package => t
    
  • 参数和值(Arguments and Values):

      symbols---一个符号[symbol]列表[list]的标识符[designator].
      package---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      export 使一个或多个在这个包 package 中可访问[accessible]的符号 symbol (不管是直接访问还是通过继承的) 变为包 package 的外部符号[external symbol].
    
      如果这些符号 symbols 中的任何一个已经是包 package 中可访问[accessible]的外部符号[external symbol], export 在那个符号[symbol]上没有效果. 如果那个符号 symbol 是作为一个内部符号出现在包 package 中, 它就被简单地改为外部状态. 如果它通过 use-package 作为一个内部符号[internal symbol]是可访问的[accessible], 它首先被导入到包 package 中, 然后再被导出[exported]. (不管接下来包 package 是否继续使用那个最初继承符号[symbol]的包[package], 这个符号 symbol 接下来都会出现[package]在包 package 中.)
    
      export 使得这里的每个符号 symbol 对于使用了包 package 的所有包[package]都是可访问的[accessible]. 所有这些包[package]都会检测命名冲突: (export s p) 为每一个在 (package-used-by-list p) 中的包执行 (find-symbol (symbol-name s) q). 注意, 在通常情况下, 一个包的最初定义期间使用 export, package-used-by-list 的结果是 nil 并且名字冲突检测需要的时间忽略不计. 当执行了多次更改时, 比如当提供给 export 这些符号 symbols 的一个列表时, 允许具体实现去单独处理每一个更改, 这样一来除了这个列表中的第一个符号以外, 其他任何符号导致的名字冲突的中止都不会取消这个列表中的第一个符号的导出. 然而, 一个由这些符号 symbols 中的其中一个导致的名字冲突错误的中止不会让那个符号[symbol]对于某些包[package]是可访问的[accessible]但对于其他是不可访问的[inaccessible]; 对于每一个被处理的符号 symbols, export 的表现就好像它是一个原子操作一样.
    
      在 export 中, 要被导出的符号 symbols 的其中一个和已经出现[present]在会继承那个新导出符号[symbol]的包[package]中的符号[symbol]之间的名字冲突, 可以通过解除捕捉另一个来选择导出的符号[symbol], 或者通过使那个已存在的符号[symbol]变为一个遮蔽符号来选择那个已存在的符号.
    
  • 示例(Examples):

    (make-package 'temp :use nil) =>  #<PACKAGE "TEMP">
    (use-package 'temp) =>  T
    (intern "TEMP-SYM" 'temp) =>  TEMP::TEMP-SYM, NIL
    (find-symbol "TEMP-SYM") =>  NIL, NIL
    (export (find-symbol "TEMP-SYM" 'temp) 'temp) =>  T
    (find-symbol "TEMP-SYM") =>  TEMP-SYM, :INHERITED
    
  • 副作用(Side Effects):

      这个包系统被修改.
    
  • 受此影响(Affected By):

      可访问[accessible]的符号[symbol].
    
  • 异常情况(Exceptional Situations):

      如果这些符号 symbols 中的任何一个在包 package 中是不可访问[accessible], 会发出一个 package-error 类型[type]的错误, 通过允许用户[user]去交互式地指定是否应该导入该符号[symbol], 这是可校正的[correctable].
    
  • 参见(See Also):

      import, unexport, 章节 11.1 (包概念)
    
  • 注意(Notes): None.

函数 FIND-SYMBOL

  • 语法(Syntax):

      find-symbol string &optional package => symbol, status
    
  • 参数和值(Arguments and Values):

      string---一个字符串[string].
      package---一个包标识符[package designator]. 默认是当前包[current package].
      symbol---在包 package 中可访问的一个符号[symbol], 或者 nil.
      status---:inherited, :external, :internal 其中之一, 或 nil.
    
  • 描述(Description):

      find-symbol 在一个包[packge]中查找名称[name]为字符串 string 的符号[symbol]. 如果在包 package 中找到名为字符串 string 的一个符号[symbol], 不管是直接地或是通过继承, 找到的符号会作为第一个值被返回; 第二个值如下:
    
      :internal
    
          如果这个符号[symbol]是作为一个内部符号[internal symbol]出现[present]在包 package 中的话.
    
      :external
    
          如果这个符号[symbol]是作为一个外部符号[external symbol]出现[present]在包 package 中的话.
    
      :inherited
    
          如果这个符号[symbol]是被包 package 通过 use-package 继承而来, 但是不是出现[present]在包 package 中的话.
    
      如果在包 package 中没有这样的可访问[accessible]的符号[symbol], 两个值都是 nil.
    
  • 示例(Examples):

    (find-symbol "NEVER-BEFORE-USED") =>  NIL, NIL
    (find-symbol "NEVER-BEFORE-USED") =>  NIL, NIL
    (intern "NEVER-BEFORE-USED") =>  NEVER-BEFORE-USED, NIL
    (intern "NEVER-BEFORE-USED") =>  NEVER-BEFORE-USED, :INTERNAL
    (find-symbol "NEVER-BEFORE-USED") =>  NEVER-BEFORE-USED, :INTERNAL
    (find-symbol "never-before-used") =>  NIL, NIL
    (find-symbol "CAR" 'common-lisp-user) =>  CAR, :INHERITED
    (find-symbol "CAR" 'common-lisp) =>  CAR, :EXTERNAL
    (find-symbol "NIL" 'common-lisp-user) =>  NIL, :INHERITED
    (find-symbol "NIL" 'common-lisp) =>  NIL, :EXTERNAL
    (find-symbol "NIL" (prog1 (make-package "JUST-TESTING" :use '())
                              (intern "NIL" "JUST-TESTING")))
    =>  JUST-TESTING::NIL, :INTERNAL
    (export 'just-testing::nil 'just-testing)
    (find-symbol "NIL" 'just-testing) =>  JUST-TESTING:NIL, :EXTERNAL
    (find-symbol "NIL" "KEYWORD")
    =>  NIL, NIL
    OR=>  :NIL, :EXTERNAL
    (find-symbol (symbol-name :nil) "KEYWORD") =>  :NIL, :EXTERNAL
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By):

      intern, import, export, use-package, unintern, unexport, unuse-package
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      intern, find-all-symbols
    
  • 注意(Notes):

      find-symbol 操作上等价于 intern, 除了它从不创建一个新的符号[symbol]. 
    

函数 FIND-PACKAGE

  • 语法(Syntax):

      find-package name => package
    
  • 参数和值(Arguments and Values):

      name---一个字符串标识符[string designator]或者一个包[package]对象[object].
      package---一个包[package]对象[object]或者 nil.
    
  • 描述(Description):

      如果名字 name 是一个字符串标识符[string designator], find-package 查找并返回这个名字或别名为 name 的包[package]. 这个查找是大小写敏感的. 如果这里没有这样的包[package], find-package 返回 nil.
    
      如果 name 是一个包[package]对象[object], 那么就返回那个包[package]对象[object].
    
  • 示例(Examples):

    (find-package 'common-lisp) =>  #<PACKAGE "COMMON-LISP">
    (find-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (find-package 'not-there) =>  NIL
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By):

      这个具体实现[implementation]创建的包[package]的集合.
    
      defpackage, delete-package, make-package, rename-package
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      make-package
    
  • 注意(Notes): None.

函数 FIND-ALL-SYMBOLS

  • 语法(Syntax):

      find-all-symbols string => symbols
    
  • 参数和值(Arguments and Values):

      string---一个字符串标识符[string designator].
      symbols---一个符号[symbol]列表[list].
    
  • 描述(Description):

      find-all-symbols 在每一个已注册包[registered package]中搜索有着和字符串 string 相同名字[name] (在 string= 下)的符号[symbol]. 返回一个所有这样符号[symbol]的列表[list]. 这个列表[list]是否排序或如何排序是依赖于具体实现的[implementation-dependent].
    
  • 示例(Examples):

    (find-all-symbols 'car)
    =>  (CAR)
    OR=>  (CAR VEHICLES:CAR)
    OR=>  (VEHICLES:CAR CAR)
    (intern "CAR" (make-package 'temp :use nil)) =>  TEMP::CAR, NIL
    (find-all-symbols 'car)
    =>  (TEMP::CAR CAR)
    OR=>  (CAR TEMP::CAR)
    OR=>  (TEMP::CAR CAR VEHICLES:CAR)
    OR=>  (CAR TEMP::CAR VEHICLES:CAR)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      find-symbol
    
  • 注意(Notes): None.

函数 IMPORT

  • 语法(Syntax):

      import symbols &optional package => t
    
  • 参数和值(Arguments and Values):

      symbols---一个符号[symbol]列表[list]的标识符[designator].
      package---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      import 添加 symbol 或 symbols 到包 package 的内部, 和出现[present]在包 package 中的或者可访问[accessible]的已存在符号[symbol]检查名字冲突. 一旦这些符号 symbols 已经被导入, 在使用 Lisp 读取器[Lisp reader]时, 它们可以在导入的包中被直接引用而不使用包前缀[package prefix].
    
      在 import 中, 这个要被导入的符号 symbol 和从某个其他包[package]中继承而来的符号之间的名字冲突, 可以通过使这个要被导入的符号 symbol 为遮蔽符号来选择这个要被导入的符号 symbol , 或者不执行这个 import 来选择已经可访问[accessible]的那个符号[symbol]来解决. 在 import 中, 和已经出现[present]在包 package 的符号[symbol]之间的名字冲突可以通过解除捕捉那个符号[symbol]或者不执行这个 import 来解决.
    
      这个导入的符号[symbol]不会被自动从这个当前包[current package]中导出, 但是如果它已经出现[present]在包中并且是外部的, 那么它是外部的这个事实不会被改变. 如果要被导入的任何符号[symbol]没有 home 包 (也就是说, (symbol-package symbol) => nil), import 会设置这个符号 symbol 的 home 包[home package]为包 package.
    
      如果这个符号 symbol 已经出现[present]在这个导入的包中, import 就没有效果.
    
  • 示例(Examples):

    (import 'common-lisp::car (make-package 'temp :use nil)) =>  T
    (find-symbol "CAR" 'temp) =>  CAR, :INTERNAL
    (find-symbol "CDR" 'temp) =>  NIL, NIL 
    
      表达式形式 (import 'editor:buffer) 取出 EDITOR 包中名为 buffer 的外部符号 (当这个表达式形式被 Lisp 读取器读取时这个符号会被查找) 并且作为一个内部符号[internal symbol]添加到当前包[current package]中. 然后这个符号 buffer 就会出现[present]在当前包[current package]中.
    
  • 副作用(Side Effects):

      这个包系统会被修改.
    
  • 受此影响(Affected By):

      这个包系统的当前状态.
    
  • 异常情况(Exceptional Situations):

      如果这些要被导入的符号 symbols 中的任何一个和包 package 中某个已经可以访问[accessible]的不同(在 eql 下)符号[symbol]有着相同[same]的名字[name] (在 string= 下), 那么 import 发出一个 package-error 类型[type]的可校正[correctable]错误, 即便这个冲突是和包 package 中的一个遮蔽符号[shadowing symbol].
    
  • 参见(See Also):

      shadow, export
    
  • 注意(Notes): None.

函数 LIST-ALL-PACKAGES

  • 语法(Syntax):

      list-all-packages <no arguments> => packages
    
  • 参数和值(Arguments and Values):

      packages---一个包[package]对象[object]列表[list].
    
  • 描述(Description):

      list-all-packages 返回所有已注册包[registered package]的新[fresh]列表[list].
    
  • 示例(Examples):

    (let ((before (list-all-packages)))
        (make-package 'temp)
        (set-difference (list-all-packages) before)) =>  (#<PACKAGE "TEMP">)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By):

      defpackage, delete-package, make-package
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also): None.

  • 注意(Notes): None.

函数 RENAME-PACKAGE

  • 语法(Syntax):

      rename-package package new-name &optional new-nicknames => package-object
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      new-name---一个包标识符[package designator].
      new-nicknames---一个字符串标识符[string designator]列表[list]. 默认是空列表[empty list].
      package-object---重命名的包 package 对象[object].
    
  • 描述(Description):

      替换包 package 的名字和别名. 包 package 的旧名字和所有旧别名会被消除并且被替换为 new-name 和 new-nicknames.
    
      如果 new-name 或任何 new-nickname 和任何已存在包名冲突, 那么后果是未定义的.
    
  • 示例(Examples):

    (make-package 'temporary :nicknames '("TEMP")) =>  #<PACKAGE "TEMPORARY">
    (rename-package 'temp 'ephemeral) =>  #<PACKAGE "EPHEMERAL">
    (package-nicknames (find-package 'ephemeral)) =>  ()
    (find-package 'temporary) =>  NIL
    (rename-package 'ephemeral 'temporary '(temp fleeting))
    =>  #<PACKAGE "TEMPORARY">
    (package-nicknames (find-package 'temp)) =>  ("TEMP" "FLEETING")
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      make-package
    
  • 注意(Notes): None.

函数 SHADOW

  • 语法(Syntax):

      shadow symbol-names &optional package => t
    
  • 参数和值(Arguments and Values):

      symbol-names---一个 string 标识符列表的标识符.
      package---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      shadow 确保带有这些符号名 symbol-names 给定的名字的那些符号[symbol]出现在包 package 中.
    
      特别地, 搜索包 package 中带有那些符号名 symbol-names 中所提供名字[name]的那些符号[symbol]. 对于每一个这样的名字[name], 如果一个对应的符号[symbol]没有出现[present]在包 package 中 (直接地, 或者通过继承), 那么会用这个名字[name]创建一个对应符号[symbol], 并且作为一个内部符号[internal symbol]插入到包 package 中. 这个对应的符号[symbol], 不管是之前存在的或是新创建的, 如果没有出现在那个包 package 中就会被添加到包 package 的遮蔽符号列表[shadowing symbols list]中.
    
  • 示例(Examples):

    (package-shadowing-symbols (make-package 'temp)) =>  NIL
    (find-symbol 'car 'temp) =>  CAR, :INHERITED
    (shadow 'car 'temp) =>  T
    (find-symbol 'car 'temp) =>  TEMP::CAR, :INTERNAL
    (package-shadowing-symbols 'temp) =>  (TEMP::CAR)
    
    (make-package 'test-1) =>  #<PACKAGE "TEST-1">
    (intern "TEST" (find-package 'test-1)) =>  TEST-1::TEST, NIL
    (shadow 'test-1::test (find-package 'test-1)) =>  T
    (shadow 'TEST (find-package 'test-1)) =>  T
    (assert (not (null (member 'test-1::test (package-shadowing-symbols
                                                (find-package 'test-1))))))
    
    (make-package 'test-2) =>  #<PACKAGE "TEST-2">
    (intern "TEST" (find-package 'test-2)) =>  TEST-2::TEST, NIL
    (export 'test-2::test (find-package 'test-2)) =>  T
    (use-package 'test-2 (find-package 'test-1))    ;should not error
    
  • 副作用(Side Effects):

      shadow 以这样一种方式更改包系统的状态, 即包一致性规则在更改期间不保持不变.
    
  • 受此影响(Affected By):

      这个包系统的当前状态.
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      package-shadowing-symbols, 章节 11.1 (包概念)
    
  • 注意(Notes):

      如果带有 symbol-names 中的一个名字的符号[symbol]已经存在于包 package 中, 但是是通过继承而来的, 这个继承的符号会被一个新创建的内部符号[internal symbol]所遮蔽[shadow[3]]. 
    

函数 SHADOWING-IMPORT

  • 语法(Syntax):

      shadowing-import symbols &optional package => t
    
  • 参数和值(Arguments and Values):

      symbols---一个符号[symbol]列表[list]的标识符[designator].
      package ---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      shadowing-import 就像是 import, 但是即便一个符号[symbol]的导入会遮蔽某个在包 package 中已经可访问[accessible]的某个符号[symbol], 它也不会发出一个错误.
    
      shadowing-import 将这些符号 symbols 中的每一个作为内部符号插入到包 package 中, 不管另一个相同名字的符号[symbol]是否被这个动作遮蔽. 如果一个相同名字的不同符号[symbol]已经出现[present]在包 package 中, 那么该符号[symbol]首先被从包 package 中解除捕捉[uninterned]. 这个新的符号[symbol]会被添加到包 package 的遮蔽符号列表中.
    
      shadowing-import 对这个范围执行名称冲突检测, 它检测一个相同名字的不同的已存在符号[symbol]是否可访问[accessible]; 如果这样, 它被这个新符号[symbol]所遮蔽, 这意味着如果它出现[present]在包 package 中它必须被解除捕捉.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (setq sym (intern "CONFLICT")) =>  CONFLICT
    (intern "CONFLICT" (make-package 'temp)) =>  TEMP::CONFLICT, NIL
    (package-shadowing-symbols 'temp) =>  NIL
    (shadowing-import sym 'temp) =>  T 
    (package-shadowing-symbols 'temp) =>  (CONFLICT)
    
  • 副作用(Side Effects):

      shadowing-import 以这样一种方式更改包系统的状态, 即一致性规则在更改期间不保持不变.
    
      包 package 的遮蔽符号列表会被修改.
    
  • 受此影响(Affected By):

      这个包系统的当前状态.
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      import, unintern, package-shadowing-symbols
    
  • 注意(Notes): None.

函数 DELETE-PACKAGE

  • 语法(Syntax):

      delete-package package => generalized-boolean
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      delete-package 从所有包系统数据结构中删除包 package. 如果这个操作成功, delete-package 返回 true, 否则就是 nil. 这个 delete-package 的效果是包 package 的名字和别名不再被识别为包名. 这个包对象[object]仍然是个包[package] (换句话说, packagep 对于它是 true 的) 但是 package-name 返回 nil. 删除这个 COMMON-LISP 包或 KEYWORD 包的后果是未定义的. 一旦包 package 被删除, 在这个包上调用任何其他包操作的后果是未指定的. 尤其, 如果在 *package* 被绑定为那个删除的包 package 或者用那个删除的包 package 作为参数的情况下调用 find-symbol, intern 和其他在一个包[package]中查找符号名的函数的后果是未指定的.
    
      如果包 package 是一个已经被删除的包[package]对象[object], delete-package 立即返回 nil.
    
      在这个操作完成后, 之前 home 包[home package]为包 package 的符号[symbol]的 home 包[home package]是依赖于具体实现的[implementation-dependent]. 除了这个, 在包 package 中可访问[accessible]的符号[symbol]不会以其他任何方式被修改; home 包[home package]不是包 package 的符号[symbol]保持不变.
    
  • 示例(Examples):

    (setq *foo-package* (make-package "FOO" :use nil))
    (setq *foo-symbol*  (intern "FOO" *foo-package*))
    (export *foo-symbol* *foo-package*)
    
    (setq *bar-package* (make-package "BAR" :use '("FOO")))
    (setq *bar-symbol*  (intern "BAR" *bar-package*))
    (export *foo-symbol* *bar-package*)
    (export *bar-symbol* *bar-package*)
    
    (setq *baz-package* (make-package "BAZ" :use '("BAR")))
    
    (symbol-package *foo-symbol*) =>  #<PACKAGE "FOO">
    (symbol-package *bar-symbol*) =>  #<PACKAGE "BAR">
    
    (prin1-to-string *foo-symbol*) =>  "FOO:FOO"
    (prin1-to-string *bar-symbol*) =>  "BAR:BAR"
    
    (find-symbol "FOO" *bar-package*) =>  FOO:FOO, :EXTERNAL
    
    (find-symbol "FOO" *baz-package*) =>  FOO:FOO, :INHERITED
    (find-symbol "BAR" *baz-package*) =>  BAR:BAR, :INHERITED
    
    (packagep *foo-package*) =>  true
    (packagep *bar-package*) =>  true
    (packagep *baz-package*) =>  true
    
    (package-name *foo-package*) =>  "FOO"
    (package-name *bar-package*) =>  "BAR"
    (package-name *baz-package*) =>  "BAZ"
    
    (package-use-list *foo-package*) =>  ()
    (package-use-list *bar-package*) =>  (#<PACKAGE "FOO">)
    (package-use-list *baz-package*) =>  (#<PACKAGE "BAR">)
    
    (package-used-by-list *foo-package*) =>  (#<PACKAGE "BAR">)
    (package-used-by-list *bar-package*) =>  (#<PACKAGE "BAZ">)
    (package-used-by-list *baz-package*) =>  ()
    
    (delete-package *bar-package*)
    >>  Error: Package BAZ uses package BAR.
    >>  If continued, BAZ will be made to unuse-package BAR,
    >>  and then BAR will be deleted.
    >>  Type :CONTINUE to continue.
    >>  Debug> :CONTINUE
    =>  T
    
    (symbol-package *foo-symbol*) =>  #<PACKAGE "FOO">
    (symbol-package *bar-symbol*) is unspecified
    
    (prin1-to-string *foo-symbol*) =>  "FOO:FOO"
    (prin1-to-string *bar-symbol*) is unspecified
    
    (find-symbol "FOO" *bar-package*) is unspecified
    
    (find-symbol "FOO" *baz-package*) =>  NIL, NIL
    (find-symbol "BAR" *baz-package*) =>  NIL, NIL
    
    (packagep *foo-package*) =>  T
    (packagep *bar-package*) =>  T
    (packagep *baz-package*) =>  T
    
    (package-name *foo-package*) =>  "FOO"
    (package-name *bar-package*) =>  NIL
    (package-name *baz-package*) =>  "BAZ"
    
    (package-use-list *foo-package*) =>  ()
    (package-use-list *bar-package*) is unspecified
    (package-use-list *baz-package*) =>  ()
    
    (package-used-by-list *foo-package*) =>  ()
    (package-used-by-list *bar-package*) is unspecified
    (package-used-by-list *baz-package*) =>  ()
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果这个 package 标识符[designator]是一个当前没有命名一个包[package]的名字, 就会发出一个 package-error 类型[type]的可校正错误. 如果尝试了校正, 就不会尝试去做删除动作; 反而, delete-package 立即返回 nil.
    
      如果包 package 被其他包[package]所使用, 就会发出一个 package-error 类型[type]的可校正[correctable]错误. 如果尝试了校正, unuse-package 会被调用来移除任何依赖, 导致包 package 的外部符号[external symbol]对于那些使用了包 package 的包[package]不再是可访问的[accessible]. delete-package 接下来删除包 package, 就好像已经没有包[package]在使用它一样.
    
  • 参见(See Also):

      unuse-package
    
  • 注意(Notes): None.

函数 MAKE-PACKAGE

  • 语法(Syntax):

      make-package package-name &key nicknames use => package
    
  • 参数和值(Arguments and Values):

      package-name---一个字符串标识符[string designator].
      nicknames---一个字符串标识符[string designator]列表[list]. 默认是空列表[empty list].
      use---一个包标识符[package designator]列表[list]. 默认是具体实现定义的[implementation-defined].
      package---一个包[package].
    
  • 描述(Description):

      创建一个名为 package-name 的新包[package].
    
      别名 nicknames 是可以被用于引用这个新的包[package]的额外的名字[name].
    
      这个 use 指定了 0 个或多个包[package], 它们的外部符号[external package]会被这个新包[package]所继承. 见函数[function] use-package.
    
  • 示例(Examples):

    (make-package 'temporary :nicknames '("TEMP" "temp")) =>  #<PACKAGE "TEMPORARY">
    (make-package "OWNER" :use '("temp")) =>  #<PACKAGE "OWNER">
    (package-used-by-list 'temp) =>  (#<PACKAGE "OWNER">)
    (package-use-list 'owner) =>  (#<PACKAGE "TEMPORARY">)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By):

      在这个系统中其他包[package]的存在.
    
  • 异常情况(Exceptional Situations):

      如果 use 指定的包[package]不存在, 那么后果是未指定的.
    
      如果包名 package-name 或这些别名 nicknames 中的任何一个已经是一个已存在的包的包名[name]或别名[nickname]那么就会发出一个可校正[correctable]错误.
    
  • 参见(See Also):

      defpackage, use-package
    
  • 注意(Notes):

      在要被使用的包[package]包含了会冲突的符号的情况下, 有必要先用 :use '() 来创建这个包, 然后使用 shadow 或 shadowing-import 来定位这些冲突, 在这之后一旦这些冲突已经被定位就使用 use-package.
    
      当要被创建的包是一个程序的静态定义的一部分而不是动态地被程序创建, 在文体上更适合使用 defpackage 而不是 make-package. 
    

宏 WITH-PACKAGE-ITERATOR

  • 语法(Syntax):

      with-package-iterator (name package-list-form &rest symbol-types) declaration* form*
      => result*
    
  • 参数和值(Arguments and Values):

      name---一个符号[symbol].
      package-list-form---一个表达式形式[form]; 求值一次来产生 package-list.
      package-list---一个包标识符[package designator]列表的标识符[designator].
      symbol-type---符号[symbol] :internal, :external, 或 :inherited 的其中之一.
      declaration---一个 declare 表达式[expression]; 不求值.
      forms---一个隐式 progn [implicit progn].
      results---这些表达式形式 forms 的值[value].
    
  • 描述(Description):

      在这个主体表达式形式 forms 的词法作用域中, 这个名字 name 是通过 macrolet 定义的, 这么一来后续的 (name) 调用会返回一个接一个来自 package-list 中包[package]的符号[symbol].
    
      从多个包[package]中继承的符号[symbol]是否会被返回不止一次是未指定的. 返回符号[symbol]的顺序没有必要反映 package-list 中包[package]的顺序. 当 package-list 有着不止一个元素时, 重复符号[symbol]会被返回一次还是不止一次是未指定的.
    
      如下的这些符号类型 symbol-types 控制一个包[package]中可访问[accessible]的哪些符号[symbol]会被返回:
    
      :internal
    
          出现[present]在包[package]中但是没有被导出[exported]的符号[symbol].
    
      :external
    
          出现[present]在包[package]中并且被导出[exported]的符号[symbol].
    
      :inherited
    
          被使用的包[package]导出[exported]并且没有被遮蔽的符号[symbol].
    
      当为符号类型 symbol-types 提供了超过一个参数时, 如果一个符号[symbol]的可访问性[accessibility]符合提供的其中一个 symbol-type, 它就会被返回. 具体实现可能通过识别额外的符号可访问性类型来扩展这个语法.
    
      一个 (name) 调用返回如下四个值:
    
      1. 表示一个符号[symbol]是否被返回的一个标志 (true 表示一个符号[symbol]被返回).
      2. 一个在指定的包[package]中是可访问[accessible]的符号[symbol].
      3. 那个符号[symbol]的可访问性类型; 换句话说, 就是符号 :internal, :external, 或 :inherited 的其中一个.
      4. 这个符号[symbol]所取自的包[package]. 这个包[package]是 package-list 中出现或命名的那些包[package]的其中一个.
    
      在所有符号[symbol]通过依次调用 (name) 已经被返回后, 然后只有一个值会被返回, 也就是 nil.
    
      第二, 第三, 和第四个值[value]的意义是, 返回的符号[symbol]在返回的包[package]中是可访问的[accessible], 其方式是由第二个返回值所指示的, 如下所示:
    
      :internal
    
          意味着出现[present]但未导出[exported].
    
      :external
    
          意味着出现[present]并导出[exported].
    
      :inherited
    
          意味着没有出现[present] (因此没有被遮蔽) 但是从某个使用的包[package]中继承.
    
      如果一个迭代的任何隐式的内部状态在这个 with-package-iterator 表达式形式的动态范围之外被返回, 比如通过在调用表达式形式[form]中返回某个闭包[closure], 那么会发生什么是不确定的.
    
      任何数量的 with-package-iterator 调用可以被嵌套, 并且最内部的那个的主体可以调用所有这些局部建立的宏[macro], 假设所有这些宏[macro]有着不同的名字.
    
  • 示例(Examples):

      下面函数应该在任何包[package]上都返回 t, 如果这个 with-package-iterator 的使用和对应 do-symbols 的使用不一致就会发出一个错误.
    
    (defun test-package-iterator (package)
      (unless (packagep package)
        (setq package (find-package package)))
      (let ((all-entries '())
            (generated-entries '()))
        (do-symbols (x package) 
          (multiple-value-bind (symbol accessibility) 
              (find-symbol (symbol-name x) package)
            (push (list symbol accessibility) all-entries)))
        (with-package-iterator (generator-fn package 
                                :internal :external :inherited)
          (loop     
            (multiple-value-bind (more? symbol accessibility pkg)
                (generator-fn)
              (unless more? (return))
              (let ((l (multiple-value-list (find-symbol (symbol-name symbol) 
                                                          package))))
                (unless (equal l (list symbol accessibility))
                  (error "Symbol ~S not found as ~S in package ~A [~S]"
                          symbol accessibility (package-name package) l))
                (push l generated-entries)))))
        (unless (and (subsetp all-entries generated-entries :test #'equal)
                      (subsetp generated-entries all-entries :test #'equal))
          (error "Generated entries and Do-Symbols entries don't correspond"))
        t))
    
      下面函数打印出每个出现[present]的符号[symbol] (可能不止一次):
    
    (defun print-all-symbols () 
      (with-package-iterator (next-symbol (list-all-packages)
                              :internal :external)
        (loop
          (multiple-value-bind (more? symbol) (next-symbol)
            (if more? 
                (print symbol)
                (return))))))
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果没有提供符号类型 symbol-types 或者提供了一个不被具体实现所识别的符号类型 symbol-type, 那么 with-package-iterator 会发出一个 program-error 类型[type]的错误.
    
      如果通过 with-package-iterator 建立的名为 name 的局部函数在它返回 false 作为它的主值[primary value]之后被调用, 那么后果是未定义的.
    
  • 参见(See Also):

      章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes): None.

函数 UNEXPORT

  • 语法(Syntax):

      unexport symbols &optional package => t
    
  • 参数和值(Arguments and Values):

      symbols---一个符号[symbol]列表[list]的标识符[designator].
      package---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      unexport 复原包 package 中的外部符号 symbols 为内部状态; 它撤销了 export 的效果.
    
      unexport 只工作在出现[present]在包 package 中的符号[symbol]上, 把它们转换回内部状态. 如果给 unexport 一个已经作为内部符号[internal symbol]出现[present]在包 package 的一个符号[symbol], 它什么都不做.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (export (intern "CONTRABAND" (make-package 'temp)) 'temp) =>  T
    (find-symbol "CONTRABAND") =>  NIL, NIL 
    (use-package 'temp) =>  T 
    (find-symbol "CONTRABAND") =>  CONTRABAND, :INHERITED
    (unexport 'contraband 'temp) =>  T
    (find-symbol "CONTRABAND") =>  NIL, NIL
    
  • 副作用(Side Effects):

      包系统会被修改.
    
  • 受此影响(Affected By):

      包系统的当前状态
    
  • 异常情况(Exceptional Situations):

      如果给 unexport 一个在包 package 中不是可访问[accessible]的符号[symbol], 就会发出一个 package-error 类型[type]的错误.
    
      如果包 package 是 KEYWORD 包或者 COMMON-LISP 包, 那么后果是未定义的.
    
  • 参见(See Also):

      export, 章节 11.1 (包概念)
    
  • 注意(Notes): None.

函数 UNINTERN

  • 语法(Syntax):

      unintern symbol &optional package => generalized-boolean
    
  • 参数和值(Arguments and Values):

      symbol---一个符号[symbol].
      package---一个包标识符[package designator]. 默认是当前包[current package].
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      unintern 从包 package 中移除符号 symbol. 如果符号 symbol 出现[present]在包 package 中, 它被从包 package 中被移除并且如果出现在包 package 的遮蔽符号列表[shadowing symbols list]中那么也从其中移除. 如果包 package 是这个符号 symbol 的 home 包[home package], 符号 symbol 会被变成没有 home 包[home package]. 符号 symbol 可能在包 package 中通过继承仍然是可访问的[accessible].
    
      unintern 的使用可以导致一个没有记录的 home 包[home package]的符号[symbol], 但是事实上在某个包[package]中是可访问的[accessible]. Common Lisp 没有检查这个异常情况, 并且这些符号[symbol]打印的前面总是有 #:.
    
      如果 unintern 移除了符号 symbol 那么它就返回 true, 否则就是 nil.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (setq temps-unpack (intern "UNPACK" (make-package 'temp))) =>  TEMP::UNPACK 
    (unintern temps-unpack 'temp) =>  T
    (find-symbol "UNPACK" 'temp) =>  NIL, NIL 
    temps-unpack =>  #:UNPACK 
    
  • 副作用(Side Effects):

      unintern 以这样一种方式更改包系统的状态, 即一致性规则在更改期间不保持不变.
    
  • 受此影响(Affected By):

      包系统的当前状态
    
  • 异常情况(Exceptional Situations):

      给 unintern 一个遮蔽符号可以揭露一个之前通过遮蔽解决的名字冲突. 如果包 A 包 B 和 C, A 包含了一个遮蔽符号 x, 并且 B 和 C 每个都包含名为 x 的外部符号, 如果这两个符号[symbol]是不同的, 那么从 A 中移除这个遮蔽的符号会揭露一个在 b:x 和 c:x 之间的名字冲突. 在这个情况中 unintern 会发出一个错误.
    
  • 参见(See Also):

      章节 11.1 (包概念)
    
  • 注意(Notes): None.

宏 IN-PACKAGE

  • 语法(Syntax):

      in-package name => package
    
  • 参数和值(Arguments and Values):

      name---一个字符串标识符[string designator]; 不求值.
      package---名为 name 的包[package].
    
  • 描述(Description):

      导致这个名为 name 的包[package]成为当前包[current package]---这也就是说, 成为这个 *package* 的值[value]. 如果不存在这样的包[package], 那么就会发出一个 package-error 类型[type]的错误.
    
      如果这个 in-package 调用作为顶层表达式形式[top level form]出现, 那么 in-package 做的所有事也都在编译时执行.
    
  • 示例(Examples): None.

  • 副作用(Side Effects):

      变量[variable] *package* 会被赋值. 如果这个 in-package 表达式形式[form]是一个顶层表达式形式[top level form], 这个赋值也会在编译时发生.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果这个指定的包[package]不存在, 就会发出一个 package-error 类型[type]的错误.
    
  • 参见(See Also):

      *package*
    
  • 注意(Notes): None.

函数 UNUSE-PACKAGE

  • 语法(Syntax):

      unuse-package packages-to-unuse &optional package => t
    
  • 参数和值(Arguments and Values):

      packages-to-unuse---一个包标识符[package designator]列表[list]的标识符[designator].
      package---一个包标识符[package designator]. 默认是当前包[current package].
    
  • 描述(Description):

      unuse-package 导致包 package 不再从包 packages-to-unuse 中继承所有外部符号[external symbol]; unuse-package 撤销 use-package 的效果. 包 packages-to-unuse 从包 package 的使用列表[use list]中被移除.
    
      任何已经被导入到包 package 中的符号[symbol]仍然会出现[present]在包 package 中.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (export (intern "SHOES" (make-package 'temp)) 'temp) =>  T
    (find-symbol "SHOES") =>  NIL, NIL
    (use-package 'temp) =>  T
    (find-symbol "SHOES") =>  SHOES, :INHERITED
    (find (find-package 'temp) (package-use-list 'common-lisp-user)) =>  #<PACKAGE "TEMP">
    (unuse-package 'temp) =>  T
    (find-symbol "SHOES") =>  NIL, NIL
    
  • 副作用(Side Effects):

      包 package 的使用列表[use list]会被修改.
    
  • 受此影响(Affected By):

      包系统的当前状态
    
  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      use-package, package-use-list
    
  • 注意(Notes): None.

函数 USE-PACKAGE

  • 语法(Syntax):

      use-package packages-to-use &optional package => t
    
  • 参数和值(Arguments and Values):

      packages-to-use---一个包标识符[package designator]列表[list]的标识符[designator]. 不能提供这个 KEYWORD 包.
      package---一个包标识符[package designator]. 默认是当前包[current package]. 这个包不能是 KEYWORD 包.
    
  • 描述(Description):

      use-package 导致包 package 去继承包列表 packages-to-use 的所有外部符号[external symbol]. 这些继承的符号[symbol]成为包 package 中可访问[accessible]的内部符号[internal symbol].
    
      如果 packages-to-use 还没有出现在包 package 的使用列表[use list]中, 那么它会被添加到包 package 的使用列表[use list]中. 在 packages-to-use 中的所有外部符号[external symbol]都成为包 package 中可访问[accessible]的内部符号[internal symbol]. use-package 不会导致任何新的符号[symbol]出现[present]在包 package 中但是通过继承会使它们变为可访问的[accessible].
    
      use-package 检测那些新导入符号和包 package 中已经可访问[accessible]的符号之间的名字冲突. 在 use-package 中的两个被包 package 从 packages-to-use 继承而来的外部符号之间的一个名字冲突可以通过导入它们中的其中一个符号到包 package 中并使它成为一个遮蔽符号进而选择该符号[symbol]来解决.
    
  • 示例(Examples):

    (export (intern "LAND-FILL" (make-package 'trash)) 'trash) =>  T
    (find-symbol "LAND-FILL" (make-package 'temp)) =>  NIL, NIL
    (package-use-list 'temp) =>  (#<PACKAGE "TEMP">)
    (use-package 'trash 'temp) =>  T
    (package-use-list 'temp) =>  (#<PACKAGE "TEMP"> #<PACKAGE "TRASH">)
    (find-symbol "LAND-FILL" 'temp) =>  TRASH:LAND-FILL, :INHERITED
    
  • 副作用(Side Effects):

      包 package 的使用列表[use list]会被修改.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      unuse-package, package-use-list, 章节 11.1 (包概念)
    
  • 注意(Notes):

      即便一个包[package] P2 已经使用[use]了包[package] P1, 仍然允许 P1 去使用[use] P2. 包[package]的使用不会被传递, 所以这个明显的循环不会导致问题. 
    

宏 DEFPACKAGE

  • 语法(Syntax):

      defpackage defined-package-name [[option]] => package
    
      option::= (:nicknames nickname*)* |  
                (:documentation string) |  
                (:use package-name*)* |  
                (:shadow {symbol-name}*)* |  
                (:shadowing-import-from package-name {symbol-name}*)* |  
                (:import-from package-name {symbol-name}*)* |  
                (:export {symbol-name}*)* |  
                (:intern {symbol-name}*)* |  
                (:size integer) 
    
  • 参数和值(Arguments and Values):

      defined-package-name---一个字符串标识符[string designator].
      package-name---一个包标识符[package designator].
      nickname---一个字符串标识符[string designator].
      symbol-name---一个字符串标识符[string designator].
      package---名为 package-name 的包[package].
    
  • 描述(Description):

      defpackage 按照指定创建一个包[package]并且返回这个包[package].
    
      如果 defined-package-name 已经引用了一个已存在的包[package], 该名字的名字到包的映射不会被改变. 如果新的定义与该包[package]的当前状态不一致, 后果是未定义的; 一个具体实现可能选择去修改这个已存在的包[package]来反映这个新的定义. 如果 defined-package-name 是一个符号[symbol], 就使用它的名字[name].
    
      标准的选项在下面描述.
    
      :nicknames
    
          给 :nicknames 的参数设置这个包[package]的别名为提供的这个名字.
    
      :documentation
    
          给 :documentation 的参数指定一个文档字符串[documentation string]; 它会作为一个文档字符串[documentation string]附加给这个包[package]. 在单独的 defpackage 表达式形式[form]中最多出现一个 :documentation 选项.
    
      :use
    
          给 :use 的参数设置名为 package-name 的包[package]会继承的那些包[package]. 如果没有提供 :use, 默认是和给 make-package 的 :use 实参[argument]一样的依赖于具体实现[implementation-dependent]的值.
    
      :shadow
    
          给 :shadow 的参数, symbol-names, 命名要被定义的包[package]中要被创建的符号[symbol]. 事实上这些符号[symbol]会被添加到遮蔽符号[symbol]的列表中, 就像是通过 shadow 一样.
    
      :shadowing-import-from
    
          由参数 symbol-names 命名的符号[symbol]会在指定的 package-name 中被查找 (涉及到一个查找, 就像是通过 find-symbol 一样). 产生的符号[symbol]会被导入到要被定义的包[package]中, 然后放置到遮蔽符号列表中, 就像是通过 shadowing-import. 符号[symbol]不会创建在除了要被定义的包以外的任何包[package]中.
    
      :import-from
    
          由参数 symbol-names 命名的符号[symbol]会在名为 package-name 的包[package]中被查找并且导入到要被定义的包[package]中. 符号[symbol]不会创建在除了要被定义的包以外的任何包[package]中.
    
      :export
    
          由参数 symbol-names 命名的那些符号[symbol]会在要被定义的包[package]中被找到或创建并导出[exported]. 这个 :export 选项和 :use 选项相互作用, 因为继承的符号[symbol]可以被使用而不是新创建一个. 这个 :export 选项和 :import-from 还有 :shadowing-import-from 选项相互作用, 因为导入的符号可以被使用而不是新创建一个. 如果给 :export 选项的一个参数是通过 use-package 作为一个 (继承的) 内部符号[internal symbol]是可访问的[accessible], 那么这个名为 symbol-name 的符号[symbol]首先被导入到这个要被定义的包[package]中, 然后从那个包[package]中导出[exported].
    
      :intern
    
          由参数 symbol-names 命名的符号[symbol]会在那个要被定义的包[package]中被找到或创建. 这个 :intern 选项和 :use 选项相互作用, 因为可以使用继承的符号[symbol]而不是新创建一个.
    
      :size
    
          给 :size 选项的参数声明了期望出现在那个包[package]中的符号[symbol]的近似数量. 这只是一个效率暗示, 可能被一个具体实现所忽略.
    
      这些选项出现在一个 defpackage 表达式形式中的顺序是不相关的. 它们被执行的顺序如下所述:
    
      1. :shadow 和 :shadowing-import-from.
      2. :use.
      3. :import-from 和 :intern.
      4. :export.
    
      首先确立遮蔽, 因为在 :use 选项被处理时它们可能是阻止虚假名字冲突的必要条件. 这个 :use 选项接下来被执行, 这样一来 :intern 和 :export 选项可以正常引用继承的符号[symbol]. 这个 :export 选项最后执行, 这样一来它可以引用任何其他选项创建的符号[symbol]; 特别地, 遮蔽符号[shadowing symbol]和导入符号[symbol]可以变为外部的.
    
      如果一个 defpackage 表达式形式[form]作为一个顶层表达式形式[top level form]出现, 这个宏[macro]在加载时正常执行的所有动作在编译时也必须执行.
    
  • 示例(Examples):

    (defpackage "MY-PACKAGE"
      (:nicknames "MYPKG" "MY-PKG")
      (:use "COMMON-LISP")
      (:shadow "CAR" "CDR")
      (:shadowing-import-from "VENDOR-COMMON-LISP"  "CONS")
      (:import-from "VENDOR-COMMON-LISP"  "GC")
      (:export "EQ" "CONS" "FROBOLA")
      )
    
    
    (defpackage my-package
      (:nicknames mypkg :MY-PKG)  ; remember Common Lisp conventions for case
      (:use common-lisp)          ; conversion on symbols
      (:shadow CAR :cdr #:cons)                              
      (:export "CONS")            ; this is the shadowed one.
      )
    
  • 受此影响(Affected By):

      已存在的包[package].
    
  • 异常情况(Exceptional Situations):

      如果提供的其中一个 :nicknames 已经引用了一个已存在的包[package], 就会发出一个 package-error 类型[type]的错误.
    
      如果 :size 或 :documentation 出现不止一次, 就会发出一个 package-error 类型[type]的错误.
    
      由于具体实现[implementation]可能允许去扩展选项 options, 如果一个出现的选项事实上不被主机的具体实现[implementation]所支持, 就会发出一个 program-error 类型[type]的错误.
    
      给选项 :shadow, :intern, :import-from, 和 :shadowing-import-from 的 symbol-name 参数集合必须是互斥的; 另外, 提供给 :export 和 :intern 的 symbol-name 参数也必须互斥. 在这个上下文中的互斥被定义为 symbol-names 中没有任何两个是 string= 的. 如果违反, 就会发出一个 program-error 类型[type]的错误.
    
      对于 :shadowing-import-from 和 :import-from 选项, 如果在由 package-name 命名的包[package]中对于参数 symbol-names 的其中一个没有可访问[accessible]的符号[symbol], 就会发出一个 program-error 类型[type]的可校正[correctable]错误[error].
    
      名字冲突错误会被 make-package, use-package, import, 和 export 的潜在调用所处理. 见章节 11.1 (包概念).
    
  • 参见(See Also):

      documentation, 章节 11.1 (包概念), 章节 3.2 (编译)
    
  • 注意(Notes):

      如果后续对 defpackage 的一个调用中(对于某个其他包[package])的 :import-from 或者 :shadowing-import-from 选项期望去找到这些可访问[accessible]的但没必要是外部的符号[symbol], 那么 :intern 选项是很有用的.
    
      建议整个包[package]定义放在一个单独的地方, 并且一个程序中所有的包[package]定义放在一个单独文件中. 这个文件在加载或编译任何其他依赖这些包[package]的东西之前被加载. 这样一个文件可以在 COMMON-LISP-USER 包中被读取, 避免任何初始化状态的问题.
    
      defpackage 不能用于创建两个 "相互递归" 的包, 比如:
    
      (defpackage my-package
        (:use common-lisp your-package)    ;requires your-package to exist first
        (:export "MY-FUN"))                
      (defpackage your-package
        (:use common-lisp)
        (:import-from my-package "MY-FUN") ;requires my-package to exist first
        (:export "MY-FUN"))
    
      然而, 没有措施可以在一个比较标准的 defpackage 的使用后阻止用户去使用一些包影响的函数去建立这样的联系, 例如 use-package, import, 和 export.
    
      这个 defpackage 的宏展开可以有效地把这些名字规范化为字符串[string], 这样一来即便一个源代码文件中的这个 defpackage 表达式形式有着随机的符号[symbol], 编译后的文件也只会包含字符串[string].
    
      依赖于具体实现[implementation-dependent]的额外选项经常接收一个由关键字[keyword]自身表示的关键字表达式形式作为列表 (keyword T) 的缩写; 这个语法在不支持它的具体实现中应该被报告为一个不识别选项. 
    

宏 DO-SYMBOLS, DO-EXTERNAL-SYMBOLS, DO-ALL-SYMBOLS

  • 语法(Syntax):

      do-symbols (var [package [result-form]]) declaration* {tag | statement}*
      => result*
    
      do-external-symbols (var [package [result-form]]) declaration* {tag | statement}*
      => result*
    
      do-all-symbols (var [result-form]) declaration* {tag | statement}*
      => result*
    
  • 参数和值(Arguments and Values):

      var---一个变量[variable]名[name]; 不求值.
      package---一个包标识符[package designator]; 求值. 在 do-symbols 和 do-external-symbols 中默认为当前包[current package].
      result-form---一个表达式形式[form]; 按如下所述求值. 默认为 nil.
      declaration---一个 declare 表达式[expression]; 不求值.
      tag---一个 go 标签[go tag]; 不求值.
      statement---一个复合表达式形式[compound form]; 按如下所述求值.
      results---如果发生了正常返回[normal return], 就是 result-form 返回的值[value], 否则如果发生一个显式返回[explicit return], 就是传递的那些值[value].
    
  • 描述(Description):

      do-symbols, do-external-symbols, 和 do-all-symbols 遍历包[package]中的符号[symbol]. 对于每一个选择的包[package]集合中的符号[symbol], 这个 var 都会被绑定为那个符号[symbol], 然后在主体中的那些语句 statements 会被执行. 当所有符号[symbol]都已经被处理时, result-form 被求值并且并且作为这个宏的值被返回.
    
      do-symbols 遍历包 package 中可访问[accessible]的符号[symbol]. 对于从多个包[package]中继承的符号[symbol], 语句 statements 可能被执行不止一次.
    
      do-all-symbols 在每个已注册的包[registered package]上遍历. do-all-symbols 不会处理每个符号[symbol], 因为一个在任何已注册的包[registered package]中都不是可访问[accessible]的符号[symbol]不会被处理. do-all-symbols 可能导致一个出现[present]在多个包[package]中的符号[symbol]被处理不止一次.
    
      do-external-symbols 遍历包 package 中的外部符号.
    
      当 result-form 被求值时, var 被绑定并且值为 nil.
    
      一个名为 nil 的隐式语句块[implicit block]在整个 do-symbols, do-external-symbols, 或 do-all-symbols 表达式形式[form]周围. return 或 return-from 可能被用于提前终止这个迭代.
    
      如果主体的执行影响了被包含在正在被迭代的包[package]的集合中的符号[symbol], 除了使用 unintern 去移除 var 当前的值的符号[symbol]之外, 后果是未定义的.
    
      对于这些宏中的每一个, 名称绑定的作用域[scope]不包括任何初始值表达式形式, 但是包括了那个可选的结果表达式形式.
    
      在主体中的任何 tag 都和 tagbody 中一样被对待.
    
  • 示例(Examples):

    (make-package 'temp :use nil) =>  #<PACKAGE "TEMP">
    (intern "SHY" 'temp) =>  TEMP::SHY, NIL ;SHY will be an internal symbol
                                            ;in the package TEMP
    (export (intern "BOLD" 'temp) 'temp)  =>  T  ;BOLD will be external  
    (let ((lst ()))
      (do-symbols (s (find-package 'temp)) (push s lst))
      lst)
    =>  (TEMP::SHY TEMP:BOLD)
    OR=>  (TEMP:BOLD TEMP::SHY)
    (let ((lst ()))
      (do-external-symbols (s (find-package 'temp) lst) (push s lst))
      lst) 
    =>  (TEMP:BOLD)
    (let ((lst ()))                                                     
      (do-all-symbols (s lst)
        (when (eq (find-package 'temp) (symbol-package s)) (push s lst)))
      lst)
    =>  (TEMP::SHY TEMP:BOLD)
    OR=>  (TEMP:BOLD TEMP::SHY)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      intern, export, 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes): None.

函数 INTERN

  • 语法(Syntax):

      intern string &optional package => symbol, status
    
  • 参数和值(Arguments and Values):

      string---一个字符串[string].
      package---一个包指示符[package designator]. 默认为当前包[current package].
      symbol---一个符号[symbol].
      status---:inherited, :external, :internal, 或者 nil 的其中之一.
    
  • 描述(Description):

      intern 把一个名为字符串 string 的符号[symbol]输入到包 package 中. 如果一个名字和字符串 string 相同的符号[symbol]在包 package 中已经是可访问[accessible]的了, 就把它返回. 如果在包 package 中没有这样的符号[symbol]可访问[accessible], 那么带有这个给定名字的新的符号[symbol]会被创建并输入到包 package 中作为一个内部符号[internal symbol], 如果包 package 是 KEYWROD 包就作为一个外部符号[external symbol]; 包 package 就成为创建的这个符号[symbol]的 home 包[home package].
    
      由 intern 返回的第一个值, symbol, 是这个被找到或者被创建的符号[symbol]. 第二个值[secondary value], status, 意义如下:
    
      :internal
    
          这个符号[symbol]被找到并且出现[present]在包 package 中作为一个内部符号[internal symbol].
    
      :external
    
          这个符号[symbol]被找到并且出现[present]在包 package 中作为一个外部符号[external symbol].
    
      :inherited
    
          这个符号[package]被找到并且是通过 use-package 继承而来 (这也意味着这个符号[symbol]是内部的).
    
      nil
    
          没有找到之前存在的符号[symbol], 所以创建一个.
    
      成为这个新符号[symbol]名字[name]的字符串[string]是给定的字符串 string 还是它的一个拷贝是依赖于具体实现的[implementation-dependent]. 在一个新符号[symbol]被创建的情况下, 一旦一个字符串[string]已经被给定作为给 intern 的 string 实参[argument], 如果后面尝试去修改这个字符串[string], 那么后果是未定义的.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (intern "Never-Before") =>  |Never-Before|, NIL
    (intern "Never-Before") =>  |Never-Before|, :INTERNAL 
    (intern "NEVER-BEFORE" "KEYWORD") =>  :NEVER-BEFORE, NIL
    (intern "NEVER-BEFORE" "KEYWORD") =>  :NEVER-BEFORE, :EXTERNAL
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      find-symbol, read, symbol, unintern, 章节 2.3.4 (符号标记)
    
  • 注意(Notes):

      如果这里已经有一个带有给定名字的可访问[accessible]符号[symbol], intern 不需要去做任何名字冲突检测, 因为它不会创建一个新符号[symbol]. 
    

函数 PACKAGE-NAME

  • 语法(Syntax):

      package-name package => name
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      name---一个字符串[string]或 nil.
    
  • 描述(Description):

      package-name 返回命名这个包 package 的字符串[string], 如果这个包 package 标识符[designator]是一个没有名字包[package]对象[object] (见函数[function] delete-package)就是 nil.
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    (package-name *package*) =>  "COMMON-LISP-USER"
    (package-name (symbol-package :test)) =>  "KEYWORD"
    (package-name (find-package 'common-lisp)) =>  "COMMON-LISP"
    
    (defvar *foo-package* (make-package "FOO"))
    (rename-package "FOO" "FOO0")
    (package-name *foo-package*) =>  "FOO0"
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 package 不是一个包标识符[package designator], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also): None.

  • 注意(Notes): None.

函数 PACKAGE-NICKNAMES

  • 语法(Syntax):

      package-nicknames package => nicknames
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      nicknames---一个字符串[string]列表[list].
    
  • 描述(Description):

      返回包 package 的别名字符串[string]列表[list], 不包括包 package 的名字.
    
  • 示例(Examples):

    (package-nicknames (make-package 'temporary
                                      :nicknames '("TEMP" "temp")))
    =>  ("temp" "TEMP") 
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 package 不是一个包标识符[package designator], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also): None.

  • 注意(Notes): None.

函数 PACKAGE-SHADOWING-SYMBOLS

  • 语法(Syntax):

      package-shadowing-symbols package => symbols
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      symbols---一个符号[symbol]列表[list].
    
  • 描述(Description):

      返回一个在包 package 中已经通过 shadow 或 shadowing-import (或等价的 defpackage 选项)被声明为遮蔽符号[shadowing symbol]的符号[symbol]列表[list]. 这个列表[list]中的所有符号[symbol]都出现[present]在这个包 package 中.
    
  • 示例(Examples):

    (package-shadowing-symbols (make-package 'temp)) =>  ()
    (shadow 'cdr 'temp) =>  T
    (package-shadowing-symbols 'temp) =>  (TEMP::CDR)
    (intern "PILL" 'temp) =>  TEMP::PILL, NIL
    (shadowing-import 'pill 'temp) =>  T
    (package-shadowing-symbols 'temp) =>  (PILL TEMP::CDR)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 package 不是一个包标识符[package designator], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      shadow, shadowing-import
    
  • 注意(Notes):

      这些符号 symbols 的列表是否是新[fresh]的依赖于具体实现[implementation-dependent]. 
    

函数 PACKAGE-USE-LIST

  • 语法(Syntax):

      package-use-list package => use-list
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      use-list---一个包[package]对象[object]的列表[list].
    
  • 描述(Description):

      返回被包 package 使用的其他包[package]的列表[list].
    
  • 示例(Examples):

    (package-use-list (make-package 'temp)) =>  (#<PACKAGE "COMMON-LISP">)
    (use-package 'common-lisp-user 'temp) =>  T
    (package-use-list 'temp) =>  (#<PACKAGE "COMMON-LISP"> #<PACKAGE "COMMON-LISP-USER">)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 package 不是一个包标识符[package designator], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      use-package, unuse-package
    
  • 注意(Notes): None.

函数 PACKAGE-USED-BY-LIST

  • 语法(Syntax):

      package-used-by-list package => used-by-list
    
  • 参数和值(Arguments and Values):

      package---一个包标识符[package designator].
      used-by-list---一个包[package]对象[object]的列表[list].
    
  • 描述(Description):

      package-used-by-list 返回其他使用包 package 的包[package]的列表[list].
    
  • 示例(Examples):

    (package-used-by-list (make-package 'temp)) =>  ()
    (make-package 'trash :use '(temp)) =>  #<PACKAGE "TRASH">
    (package-used-by-list 'temp) =>  (#<PACKAGE "TRASH">)
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 package 不是一个包标识符[package designator], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      use-package, unuse-package
    
  • 注意(Notes): None.

函数 PACKAGEP

  • 语法(Syntax):

      packagep object => generalized-boolean
    
  • 参数和值(Arguments and Values):

      object---一个对象[object].
      generalized-boolean---一个广义的 boolean [generalized boolean].
    
  • 描述(Description):

      如果对象 object 是 package 类型[type]的就返回 true; 否则, 返回 false.
    
  • 示例(Examples):

    (packagep *package*) =>  true 
    (packagep 'common-lisp) =>  false 
    (packagep (find-package 'common-lisp)) =>  true 
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also): None.

  • 注意(Notes):

      (packagep object) ==  (typep object 'package)
    

变量 PACKAGE

  • 值类型(Value Type):

      一个包[package]对象[object].
    
  • 初始值(Initial Value):

      COMMON-LISP-USER 包.
    
  • 描述(Description):

      不管哪个包[package]对象[object]当前是 *package* 的值[value], 都会被引用作为当前包[current package].
    
  • 示例(Examples):

    (in-package "COMMON-LISP-USER") =>  #<PACKAGE "COMMON-LISP-USER">
    *package* =>  #<PACKAGE "COMMON-LISP-USER">
    (make-package "SAMPLE-PACKAGE" :use '("COMMON-LISP"))
    =>  #<PACKAGE "SAMPLE-PACKAGE">
    (list 
      (symbol-package
        (let ((*package* (find-package 'sample-package)))
          (setq *some-symbol* (read-from-string "just-testing"))))
      *package*)
    =>  (#<PACKAGE "SAMPLE-PACKAGE"> #<PACKAGE "COMMON-LISP-USER">)
    (list (symbol-package (read-from-string "just-testing"))
          *package*)
    =>  (#<PACKAGE "COMMON-LISP-USER"> #<PACKAGE "COMMON-LISP-USER">)
    (eq 'foo (intern "FOO")) =>  true
    (eq 'foo (let ((*package* (find-package 'sample-package)))
                (intern "FOO")))
    =>  false
    
  • 受此影响(Affected By):

      load, compile-file, in-package
    
  • 参见(See Also):

      compile-file, in-package, load, package
    
  • 注意(Notes): None.

状况类型 PACKAGE-ERROR

  • 类优先级列表(Class Precedence List):

      package-error, error, serious-condition, condition, t
    
  • 描述(Description):

      这个 package-error 类型[type]由在包[package]上的操作相关的错误[error]状况[condition]组成. 这个违规的包[package] (或者包[package]名[name]) 是通过给 make-condition 的 :package 初始化参数来初始化的, 并且可以通过函数[function] package-error-package 来访问.
    
  • 参见(See Also):

      package-error-package, 章节 9 (状况) 
    

函数 PACKAGE-ERROR-PACKAGE

  • 语法(Syntax):

      package-error-package condition => package
    
  • 参数和值(Arguments and Values):

      condition---一个 package-error 类型[type]的状况[condition].
      package---一个包标识符[package designator].
    
  • 描述(Description):

      返回这个状况 condition 表示的情况[situation]中出问题的包[package]的标识符[designator].
    
  • 示例(Examples):

    (package-error-package 
      (make-condition 'package-error
        :package (find-package "COMMON-LISP")))
    =>  #<Package "COMMON-LISP">
    
  • 副作用(Side Effects): None.

  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations): None.

  • 参见(See Also):

      package-error
    
  • 注意(Notes): None.

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值