Bootstrap

Common Lisp 超规范(译文):14.构造

14. 构造(Conses)

14.1 构造的概念

一个构造[cons]是一个有着称为 car 和 cdr 两部分的复合数据对象[object].

car  cons    rplacd  
cdr  rplaca          

Figure 14-1. 和构造相关的已定义的名字.

根据上下文, 一组连接的构造[cons]可以以各种不同的视角来看. 提供了各种各样的操作来支持这些不同的视角.

14.1.1 构造作为树

一个树[tree]是一个由构造[cons]和原子[atom]组成的二元递归数据结构: 这些构造[cons]自身也是树[true] (有时称之为 “子树(subtree)” 或 “分枝(branch)”), 而这些原子[atom]是终止节点 (有时也称为叶[leave]). 典型地, 这些叶[leave]表示数据而分枝确定这些数据的关系.

caaaar  caddar  cdar       nsubst         
caaadr  cadddr  cddaar     nsubst-if      
caaar   caddr   cddadr     nsubst-if-not  
caadar  cadr    cddar      nthcdr         
caaddr  cdaaar  cdddar     sublis         
caadr   cdaadr  cddddr     subst          
caar    cdaar   cdddr      subst-if       
cadaar  cdadar  cddr       subst-if-not   
cadadr  cdaddr  copy-tree  tree-equal     
cadar   cdadr   nsublis                   

Figure 14-2. 和树相关的一些已定义的名字.
14.1.1.1 对必须是树的参数的常规限制

除非有明确的声明, 对于任何接受一个必须为树[tree]的形参[parameter]的标准化[standardized]函数[function], 如果该树[tree]是环状的, 其后果是未定义的.

14.1.2 构造作为列表

一个列表[list]是一个构造[cons]的链, 其中每一个构造[cons]的 car 是这个列表[list]的元素[element], 而每个构造[cons]的 cdr 是这个链中的下一个链接或一个终止原子[atom].

一个正规列表[proper list]列表是一个以空列表[empty list]终止的列表[list]. 这个空列表[empty list]是一个正规列表[proper list], 但不是一个构造[cons].

一个非正规列表[improper list]是一个不是正规列表[proper list]的列表[list]; 这也就是说, 它是一个环状列表[circular list]或一个点列表[dotted list].

一个点列表[dotted list]是一个有着一个不是空列表[empty list]的终止原子[atom]的列表[list]. 一个非 nil [non-nil]的原子[atom]自身不会被当作任何种类的列表[list]—甚至不是一个点列表[dotted list].

一个环状列表[circular list]是一个构造[cons]的链, 由于这个链中的某个构造[cons]是后面一个构造[cons]的 cdr 所以它没有终止.

append      last           nbutlast  rest       
butlast     ldiff          nconc     revappend  
copy-alist  list           ninth     second     
copy-list   list*          nreconc   seventh    
eighth      list-length    nth       sixth      
endp        make-list      nthcdr    tailp      
fifth       member         pop       tenth      
first       member-if      push      third      
fourth      member-if-not  pushnew              

Figure 14-3. 和列表相关的已定义的名字.
14.1.2.1 列表作为关联列表

一个关联列表[association list]是一个表示一个键[key]和值[value]关联的构造[cons]的列表[list], 其中每一个构造[cons]的 car 是那个键[key]而 cdr 是和那个键[key]关联的值[value].

acons  assoc-if      pairlis  rassoc-if      
assoc  assoc-if-not  rassoc   rassoc-if-not  

Figure 14-4. 和关联列表相关的已定义的名字.
14.1.2.2 列表作为集合

列表[list]有时可以通过把它们的元素当作无序的并且假定这里没有重复元素来视作集合.

adjoin         nset-difference    set-difference    union  
intersection   nset-exclusive-or  set-exclusive-or         
nintersection  nunion             subsetp                  

Figure 14-5. 和集合相关的已定义的名字.
14.1.2.3 对必须是列表的参数的普通限制

除非有明确的声明, 对于任何接受一个需要为列表[list]的形参[parameter]的标准化[standard]函数[function], 如果接收到的值[value]是一个点列表[dotted list], 都应该准备发出一个 type-error 类型[type]的错误.

除非有明确的声明, 对于任何接受一个需要为列表[list]的参数[parameter]的标准化[standard]函数[function], 如果列表[list]是环状[circular]的那么后果是未定义的.

14.2 构造的字典

系统类 LIST

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

      list, sequence, t
    
  • 描述(Description):

      一个列表[list]是一个构造[cons]的链, 其中每一个构造[cons]的 car 是这个列表[list]的一个元素[element], 而每个构造[cons]的 cdr 是这个链的下一个链接或一个终止原子[atom].
    
      一个正规列表[proper list]是一个由空列表[empty list] () 来终止的构造[cons]链, 这个空列表[empty list]自身也是一个正规列表[proper list]. 一个点列表[dotted list]是一个终止原子[atom]不是空列表[empty list]的列表[list]. 一个环状列表[circular list]是一个由于链中的某个构造[cons]是后面的构造[cons]的 cdr 而没有终止的 cons 链.
    
      点列表[dotted list]和环状列表[circular list]也是列表[list], 但通常在这个规范中的那个非限制术语 "list" 意味着正规列表[proper list]. 然而, 类型[type] list 明确地包含了点列表[dotted list]和环状列表[circular list].
    
      对于一个列表[list]中的每个元素[element]这里都有一个构造[cons]. 空列表[empty list]没有元素[element]并且不是一个构造[cons].
    
      类型[type] cons 和 null 构成 list 类型[type]的详尽分区[exhaustive partition].
    
  • 参见(See Also):

      章节 2.4.1 (左圆括号), 章节 22.1.3.5 (打印列表和构造(cons)) 
    

系统类 NULL

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

      null, symbol, list, sequence, t
    
  • 描述(Description):

      null 类型[type]仅有的对象[object]是 nil, 它表示空列表[empty list]并且也可以被标记为 ().
    
  • 参见(See Also):

      章节 2.3.4 (符号标记), 章节 2.4.1 (左圆括号), 章节 22.1.3.3 (打印符号) 
    

系统类 CONS

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

      cons, list, sequence, t
    
  • 描述(Description):

      一个 cons 是一个有着两个部分的复合对象[object], 这两个部分称为 car 和 cdr. 这些组成了一个点对[dotted pair]. 每个部分可以是任何对象[object].
    
  • 复合类型指定符类别(Compound Type Specifier Kind):

      详细的.
    
  • 复合类型指定符语法(Compound Type Specifier Syntax):

      cons [car-typespec [cdr-typespec]]
    
  • 复合类型指定符参数(Compound Type Specifier Arguments):

      car-typespec---一个类型指定符[type specifier], 或者符号[symbol] *. 默认是符号[symbol] *.
    
      cdr-typespec---一个类型指定符[type specifier], 或者符号[symbol] *. 默认是符号[symbol] *.
    
  • 复合类型指定符描述(Compound Type Specifier Description):

      这个表示这个 car 受类型[type] car-typespec 约束而 cdr 受类型[type] cdr-typespec 约束的 cons 集合. (如果 car-typespec 或 cdr-typespec 是 *, 它就好像被表示为类型[type] t.)
    
  • 参见(See Also):

      章节 2.4.1 (左圆括号), 章节 22.1.3.5 (打印列表和构造(cons)) 
    

类型 ATOM

  • 超类型(Supertypes):

      atom, t
    
  • 描述(Description):

      它等价于 (not cons). 
    

函数 CONS

  • 语法(Syntax):

      cons object-1 object-2 => cons
    
  • 参数和值(Arguments and Values):

      object-1---一个对象[object].
      object-2---一个对象[object].
      cons---一个 cons.
    
  • 描述(Description):

      创建一个新[fresh]的 cons, 它的 car 是对象 object-1 而它的 cdr 是对象 object-2.
    
  • 示例(Examples):

    (cons 1 2) =>  (1 . 2)
    (cons 1 nil) =>  (1)
    (cons nil 2) =>  (NIL . 2)
    (cons nil nil) =>  (NIL)
    (cons 1 (cons 2 (cons 3 (cons 4 nil)))) =>  (1 2 3 4)
    (cons 'a 'b) =>  (A . B)
    (cons 'a (cons 'b (cons 'c '()))) =>  (A B C)
    (cons 'a '(b c d)) =>  (A B C D)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      list
    
  • 注意(Notes):

      如果 object-2 是一个列表[list], cons 可以被认为产生一个和 object-2 相似但是前面加上对象 object-1 的新的列表[list]. 
    

函数 CONSP

  • 语法(Syntax):

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

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

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

    (consp nil) =>  false
    (consp (cons 1 2)) =>  true
    
      空列表[empty list]不是一个 cons, 因此
    
    (consp '()) ==  (consp 'nil) =>  false
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      listp
    
  • 注意(Notes):

      (consp object) ==  (typep object 'cons) ==  (not (typep object 'atom)) ==  (typep object '(not atom))
    

函数 ATOM

  • 语法(Syntax):

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

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

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

    (atom 'sss) =>  true
    (atom (cons 1 2)) =>  false
    (atom nil) =>  true
    (atom '()) =>  true
    (atom 3) =>  true
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also): None.

  • 注意(Notes):

      (atom object) ==  (typep object 'atom) ==  (not (consp object))
      ==  (not (typep object 'cons)) ==  (typep object '(not cons))
    

函数 RPLACA, RPLACD

  • 语法(Syntax):

      rplaca cons object => cons
    
      rplacd cons object => cons
    
  • 发音(Pronunciation):

      rplaca: [,ree'plakuh] or [,ruh'plakuh]
    
      rplacd: [,ree'plakduh] or [,ruh'plakduh] or [,ree'plakdee] or [,ruh'plakdee]
    
  • 参数和值(Arguments and Values):

      cons---一个 cons.
      object---一个对象[object].
    
  • 描述(Description):

      rplaca 替换这个 cons 的 car 为对象 object.
    
      rplacd 替换这个 cons 的 cdr 为对象 object.
    
  • 示例(Examples):

    (defparameter *some-list* (list* 'one 'two 'three 'four)) =>  *some-list*
    *some-list* =>  (ONE TWO THREE . FOUR)
    (rplaca *some-list* 'uno) =>  (UNO TWO THREE . FOUR)
    *some-list* =>  (UNO TWO THREE . FOUR)
    (rplacd (last *some-list*) (list 'IV)) =>  (THREE IV)
    *some-list* =>  (UNO TWO THREE IV)
    
  • 副作用(Side Effects):

      这个 cons 被修改.
    
  • 受此影响(Affected By): None.

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

      如果 cons 不是一个构造(cons) 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also): None.

  • 注意(Notes): None.

访问器 CAR, CDR, CAAR, CADR, CDAR, CDDR, CAAAR, CAADR, CADAR, CADDR, CDAAR, CDADR, CDDAR, CDDDR, CAAAAR, CAAADR, CAADAR, CAADDR, CADAAR, CADADR, CADDAR, CADDDR, CDAAAR, CDAADR, CDADAR, CDADDR, CDDAAR, CDDADR, CDDDAR, CDDDDR

  • 语法(Syntax):

      car x => object
    
      cdr x => object
    
      caar x => object
    
      cadr x => object
    
      cdar x => object
    
      cddr x => object
    
      caaar x => object
    
      caadr x => object
    
      cadar x => object
    
      caddr x => object
    
      cdaar x => object
    
      cdadr x => object
    
      cddar x => object
    
      cdddr x => object
    
      caaaar x => object
    
      caaadr x => object
    
      caadar x => object
    
      caaddr x => object
    
      cadaar x => object
    
      cadadr x => object
    
      caddar x => object
    
      cadddr x => object
    
      cdaaar x => object
    
      cdaadr x => object
    
      cdadar x => object
    
      cdaddr x => object
    
      cddaar x => object
    
      cddadr x => object
    
      cdddar x => object
    
      cddddr x => object
    
      (setf (car x) new-object)
    
      (setf (cdr x) new-object)
    
      (setf (caar x) new-object)
    
      (setf (cadr x) new-object)
    
      (setf (cdar x) new-object)
    
      (setf (cddr x) new-object)
    
      (setf (caaar x) new-object)
    
      (setf (caadr x) new-object)
    
      (setf (cadar x) new-object)
    
      (setf (caddr x) new-object)
    
      (setf (cdaar x) new-object)
    
      (setf (cdadr x) new-object)
    
      (setf (cddar x) new-object)
    
      (setf (cdddr x) new-object)
    
      (setf (caaaar x) new-object)
    
      (setf (caaadr x) new-object)
    
      (setf (caadar x) new-object)
    
      (setf (caaddr x) new-object)
    
      (setf (cadaar x) new-object)
    
      (setf (cadadr x) new-object)
    
      (setf (caddar x) new-object)
    
      (setf (cadddr x) new-object)
    
      (setf (cdaaar x) new-object)
    
      (setf (cdaadr x) new-object)
    
      (setf (cdadar x) new-object)
    
      (setf (cdaddr x) new-object)
    
      (setf (cddaar x) new-object)
    
      (setf (cddadr x) new-object)
    
      (setf (cdddar x) new-object)
    
      (setf (cddddr x) new-object)
    
  • 发音(Pronunciation):

      cadr: ['ka,duhr]
    
      caddr: ['kaduh,duhr] or ['ka,dduhr]
    
      cdr: ['k,duhr]
    
      cddr: ['kduh,duhr] or ['kuh,dduhr]
    
  • 参数和值(Arguments and Values):

      x---一个列表[list].
      object---一个对象[object].
      new-object---一个对象[object].
    
  • 描述(Description):

      如果 x 是一个 cons, car 返回这个 cons 的 car. 如果 x 是 nil, car 返回 nil.
    
      如果 x 是一个 cons, cdr 返回这个 cons 的 cdr. 如果 x 是 nil, cdr 返回 nil.
    
      提供了执行多达四个 car 和 cdr 操作组合的函数. 它们的名字[name]有一个 C, 后面跟着 2, 3, 或 4 个 A 或 D, 最后是一个 R. 在每个函数[function]名字[name]中的 A 和 D 的序列被选择用来确定这个函数执行的 car 和 cdr 操作的序列. 这个 A 和 D 出现的顺序是对应操作被执行的顺序的倒序. 下一段准确地定义了这些关系.
    
          这个位置 ...     相当于这个位置 ...  
          (caar x)        (car (car x))                    
          (cadr x)        (car (cdr x))                    
          (cdar x)        (cdr (car x))                    
          (cddr x)        (cdr (cdr x))                    
          (caaar x)       (car (car (car x)))              
          (caadr x)       (car (car (cdr x)))              
          (cadar x)       (car (cdr (car x)))              
          (caddr x)       (car (cdr (cdr x)))              
          (cdaar x)       (cdr (car (car x)))              
          (cdadr x)       (cdr (car (cdr x)))              
          (cddar x)       (cdr (cdr (car x)))              
          (cdddr x)       (cdr (cdr (cdr x)))              
          (caaaar x)      (car (car (car (car x))))        
          (caaadr x)      (car (car (car (cdr x))))        
          (caadar x)      (car (car (cdr (car x))))        
          (caaddr x)      (car (car (cdr (cdr x))))        
          (cadaar x)      (car (cdr (car (car x))))        
          (cadadr x)      (car (cdr (car (cdr x))))        
          (caddar x)      (car (cdr (cdr (car x))))        
          (cadddr x)      (car (cdr (cdr (cdr x))))        
          (cdaaar x)      (cdr (car (car (car x))))        
          (cdaadr x)      (cdr (car (car (cdr x))))        
          (cdadar x)      (cdr (car (cdr (car x))))        
          (cdaddr x)      (cdr (car (cdr (cdr x))))        
          (cddaar x)      (cdr (cdr (car (car x))))        
          (cddadr x)      (cdr (cdr (car (cdr x))))        
          (cdddar x)      (cdr (cdr (cdr (car x))))        
          (cddddr x)      (cdr (cdr (cdr (cdr x))))        
    
          Figure 14-6. CAR 和 CDR 变体
    
      setf 也可以和这些函数中的任意一个一起使用来改变一个已存在的 x 的成分, 但是 setf 不会创建新的成分. 所以, 比如, 一个 cons 的 car 可以用 car 的 setf 来赋值, 但是 nil 的 car 不能使用 car 的 setf 来赋值. 相似地, 一个 car 为一个 cons 的 cons, 它的 car 的 car 可以使用 caar 的 setf 来赋值, 但是 nil 和一个 car 为 nil 的 cons 不能使用 caar 的 setf 来赋值.
    
      参数 x 允许为一个点列表[dotted list]或者一个环状列表[circular list].
    
  • 示例(Examples):

    (car nil) =>  NIL  
    (cdr '(1 . 2)) =>  2
    (cdr '(1 2)) =>  (2)
    (cadr '(1 2)) =>  2 
    (car '(a b c)) =>  A
    (cdr '(a b c)) =>  (B C)
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果函数 car 和 cdr 收到参数不是一个列表[list], 那么它应该发出一个 type-error 类型的错误. 其他函数 (caar, cadr, ... cddddr) 应该处于错误检查的目的而表现地就好像是通过 car 和 cdr 的适当调用来定义的.
    
  • 参见(See Also):

      rplaca, first, rest
    
  • 注意(Notes):

      一个 cons 的 car 也可以通过使用 rplaca 来修改, 而一个 cons 的 cdr 可以使用 rplacd 来修改.
    
      (car x)    ==  (first x)
      (cadr x)   ==  (second x) ==  (car (cdr x))
      (caddr x)  ==  (third x)  ==  (car (cdr (cdr x)))
      (cadddr x) ==  (fourth x) ==  (car (cdr (cdr (cdr x))))
    

函数 COPY-TREE

  • 语法(Syntax):

      copy-tree tree => new-tree
    
  • 参数和值(Arguments and Values):

      tree---一个树[tree].
      new-tree---一个树[tree].
    
  • 描述(Description):

      创建一个 cons 树[tree]的一个拷贝[copy].
    
      如果 tree 不是一个 cons, 它会被返回; 否则, 结果就是在树 tree 的 car 和 cdr 上调用 copy-tree 的结果的 cons. 换句话说, 由 tree 表示的树中的所有 cons 会被递归复制, 只有在没有 cons 时停止.
    
      copy-tree 不保持环状和子结构的共享.
    
  • 示例(Examples):

    (setq object (list (cons 1 "one")
                        (cons 2 (list 'a 'b 'c))))
    =>  ((1 . "one") (2 A B C))
    (setq object-too object) =>  ((1 . "one") (2 A B C))
    (setq copy-as-list (copy-list object))
    (setq copy-as-alist (copy-alist object))
    (setq copy-as-tree (copy-tree object))
    (eq object object-too) =>  true
    (eq copy-as-tree object) =>  false
    (eql copy-as-tree object) =>  false
    (equal copy-as-tree object) =>  true
    (setf (first (cdr (second object))) "a"
          (car (second object)) "two"
          (car object) '(one . 1)) =>  (ONE . 1)
    object =>  ((ONE . 1) ("two" "a" B C))
    object-too =>  ((ONE . 1) ("two" "a" B C))
    copy-as-list =>  ((1 . "one") ("two" "a" B C))
    copy-as-alist =>  ((1 . "one") (2 "a" B C))
    copy-as-tree =>  ((1 . "one") (2 A B C)) 
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      tree-equal
    
  • 注意(Notes): None.

函数 SUBLIS, NSUBLIS

  • 语法(Syntax):

      sublis alist tree &key key test test-not => new-tree
    
      nsublis alist tree &key key test test-not => new-tree
    
  • 参数和值(Arguments and Values):

      alist---一个关联列表[association list].
      tree---一个树[tree].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单实参[argument]函数[function]的标识符[designator], 或者 nil.
      new-tree---一个树[tree].
    
  • 描述(Description):

      sublis 对树 tree (一个 cons 结构) 中的对象[object]进行替换. nsublis 和 sublis 相似但是它会破坏性修改树 tree 中相关的结构.
    
      sublis 查找树 tree 中的所有子树和叶; 如果一个子树或叶作为关联列表 alist 中的一个键出现 (这也就是说, 这个键和这个子树或叶满足这个测试条件[satisfy the test]), 那么它就会被和那个键关联的对象[object]所替换. 这个操作是非破坏性的. 实际上, sublis 可以同时执行多个 subst 操作.
    
      如果 sublis 成功了, 一个树 tree 的新的拷贝会被返回, 其中每个出现的这样的子树或叶都会被它关联的对象[object]所替代. 如果没有做出改变, 返回原始的树. 原始的树保持不修改, 但是产生的树可能和它共享存储格(cell).
    
      nsublis 允许去修改树 tree, 否则就返回和 sublis 一样的值.
    
  • 示例(Examples):

    (sublis '((x . 100) (z . zprime))
            '(plus x (minus g z x p) 4 . x))
    =>  (PLUS 100 (MINUS G ZPRIME 100 P) 4 . 100)
    (sublis '(((+ x y) . (- x y)) ((- x y) . (+ x y)))
            '(* (/ (+ x y) (+ x p)) (- x y))
            :test #'equal)
    =>  (* (/ (- X Y) (+ X P)) (+ X Y))
    (setq tree1 '(1 (1 2) ((1 2 3)) (((1 2 3 4)))))
    =>  (1 (1 2) ((1 2 3)) (((1 2 3 4))))
    (sublis '((3 . "three")) tree1) 
    =>  (1 (1 2) ((1 2 "three")) (((1 2 "three" 4))))
    (sublis '((t . "string"))
              (sublis '((1 . "") (4 . 44)) tree1)
              :key #'stringp)
    =>  ("string" ("string" 2) (("string" 2 3)) ((("string" 2 3 44))))
    tree1 =>  (1 (1 2) ((1 2 3)) (((1 2 3 4))))
    (setq tree2 '("one" ("one" "two") (("one" "Two" "three"))))
    =>  ("one" ("one" "two") (("one" "Two" "three"))) 
    (sublis '(("two" . 2)) tree2) 
    =>  ("one" ("one" "two") (("one" "Two" "three"))) 
    tree2 =>  ("one" ("one" "two") (("one" "Two" "three"))) 
    (sublis '(("two" . 2)) tree2 :test 'equal) 
    =>  ("one" ("one" 2) (("one" "Two" "three"))) 
    
    (nsublis '((t . 'temp))
              tree1
              :key #'(lambda (x) (or (atom x) (< (list-length x) 3))))
    =>  ((QUOTE TEMP) (QUOTE TEMP) QUOTE TEMP) 
    
  • 副作用(Side Effects):

      nsublis 修改树 tree.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      subst, 章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数被废弃了.
    
      由于那些副作用变体 (例如, nsublis) 潜在地改变了它要经过的路径, 它们对共享或环状结构的影响可能会以令人惊讶的方式不同于它们的无副作用替代方案. 为了看到这个, 细想下面这个带副作用的行为, 它可能被某些具体实现展示:
    
    (defun test-it (fn)
      (let* ((shared-piece (list 'a 'b))
              (data (list shared-piece shared-piece)))
        (funcall fn '((a . b) (b . a)) data)))
    (test-it #'sublis) =>  ((B A) (B A))
    (test-it #'nsublis) =>  ((A B) (A B))
    

函数 SUBST, SUBST-IF, SUBST-IF-NOT, NSUBST, NSUBST-IF, NSUBST-IF-NOT

  • 语法(Syntax):

      subst new old tree &key key test test-not => new-tree
    
      subst-if new predicate tree &key key => new-tree
    
      subst-if-not new predicate tree &key key => new-tree
    
      nsubst new old tree &key key test test-not => new-tree
    
      nsubst-if new predicate tree &key key => new-tree
    
      nsubst-if-not new predicate tree &key key => new-tree
    
  • 参数和值(Arguments and Values):

      new---一个对象[object].
      old---一个对象[object].
      predicate---命名一个函数[function]的符号[symbol]或者一个返回广义 boolean [generalized boolean]值的单参数函数[function].
      tree---一个树[tree].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      new-tree---一个树.
    
  • 描述(Description):

      subst, subst-if, 和 subst-if-not 在树 tree 上执行替换操作. 每个函数都搜索树 tree, 以查找满足测试条件[satisfy the test]的某个元素或子表达式的某个旧项 old 的出现.
    
      nsubst, nsubst-if, 和 nsubst-if-not 分别类似于 subst, subst-if, 和 subst-if-not, 除了原始的树 tree 会被修改.
    
      subst 做一份树 tree 的拷贝, 用新的 new 替换树 tree 中每一个旧值为 old 和满足测试条件[satisfy the test]的子树和叶 (不管那个子树和叶是它的父节点的 car 还是 cdr).
    
      nsubst 是 subst 的一个破坏性版本. 树 tree 的列表结构会被破坏性地替换, 用新值 new 替换树 tree 中的旧值为 old 和满足测试条件 test 的叶.
    
      对于 subst, subst-if, 和 subst-if-not, 如果这些函数成功了, 这个树 tree 的一个新的拷贝会被返回, 其中每一个这样的元素的出现都被替换成那个新的元素 new 或子表达式. 如果没有发生变化, 返回原始的树 tree. 原始的树 tree 保持不变, 但是产生的树可能和它共享存储.
    
      对于 nsubst, nsubst-if, 和 nsubst-if-not 原始的树 tree 会被修改并作为这个函数的结果返回, 但是结果可能和树 tree 不是 eq 的.
    
  • 示例(Examples):

    (setq tree1 '(1 (1 2) (1 2 3) (1 2 3 4))) =>  (1 (1 2) (1 2 3) (1 2 3 4))
    (subst "two" 2 tree1) =>  (1 (1 "two") (1 "two" 3) (1 "two" 3 4))
    (subst "five" 5 tree1) =>  (1 (1 2) (1 2 3) (1 2 3 4))
    (eq tree1 (subst "five" 5 tree1)) =>  implementation-dependent
    (subst 'tempest 'hurricane
            '(shakespeare wrote (the hurricane)))
    =>  (SHAKESPEARE WROTE (THE TEMPESTSUBST))
    (subst 'foo 'nil '(shakespeare wrote (twelfth night)))
    =>  (SHAKESPEARE WROTE (TWELFTH NIGHT . FOO) . FOO)
    (subst '(a . cons) '(old . pair)
            '((old . spice) ((old . shoes) old . pair) (old . pair))
            :test #'equal)
    =>  ((OLD . SPICE) ((OLD . SHOES) A . CONS) (A . CONS))
    
    (subst-if 5 #'listp tree1) =>  5
    (subst-if-not '(x) #'consp tree1) 
    =>  (1 X)
    
    tree1 =>  (1 (1 2) (1 2 3) (1 2 3 4))
    (nsubst 'x 3 tree1 :key #'(lambda (y) (and (listp y) (third y)))) 
    =>  (1 (1 2) X X)
    tree1 =>  (1 (1 2) X X)
    
  • 副作用(Side Effects):

      nsubst, nsubst-if, 和 nsubst-if-not 可能修改树 tree 的树结构[tree structure].
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      substitute, nsubstitute, 章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数被废弃了.
    
      函数 subst-if-not 和 nsubst-if-not 被废弃了.
    
      subst 的一个可能的定义:
    
    (defun subst (old new tree &rest x &key test test-not key)
      (cond ((satisfies-the-test old tree :test test
                                  :test-not test-not :key key)
              new)
            ((atom tree) tree)
            (t (let ((a (apply #'subst old new (car tree) x))
                      (d (apply #'subst old new (cdr tree) x)))
                  (if (and (eql a (car tree))
                          (eql d (cdr tree)))
                      tree
                      (cons a d))))))
    

函数 TREE-EQUAL

  • 语法(Syntax):

      tree-equal tree-1 tree-2 &key test test-not => generalized-boolean
    
  • 参数和值(Arguments and Values):

      tree-1---一个树[tree].
      tree-2---一个树[tree].
      test---一个返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---一个返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      tree-equal 测试两个树是否是相同的形状并且有着相同的叶. 如果 tree-1 和 tree-2 都是原子[atom]并且满足测试条件[satisfy the test], 或者它们都是 cons 并且 tree-1 的 car 和 tree-2 的 car 是 tree-equal 的而 tree-1 的 cdr 和 tree-2 的 cdr 也是 tree-equal 的, 那么 tree-equal 返回 true. 否则, tree-equal 返回 false.
    
      tree-equal 递归地比较 cons 而不是其他有成分的对象[object].
    
      给 :test 或 :test-not 函数的第一个参数是 tree-1 或者 tree-1 的一个 car 或 cdr; 第二个参数是 tree-2 或 tree-2 的一个 car 或 cdr.
    
  • 示例(Examples):

    (setq tree1 '(1 (1 2))
          tree2 '(1 (1 2))) =>  (1 (1 2))
    (tree-equal tree1 tree2) =>  true
    (eql tree1 tree2) =>  false
    (setq tree1 '('a ('b 'c))
          tree2 '('a ('b 'c))) =>  ('a ('b 'c)) 
    =>  ((QUOTE A) ((QUOTE B) (QUOTE C)))
    (tree-equal tree1 tree2 :test 'eq) =>  true
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 tree-1 和 tree-2 都是环状的那么后果是不确定的.
    
  • 参见(See Also):

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

      这个 :test-not 参数被废弃了. 
    

函数 COPY-LIST

  • 语法(Syntax):

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

      list---一个正规列表[proper list]或者一个点列表[dotted list].
      copy---一个列表[list].
    
  • 描述(Description):

      返回列表 list 的一个拷贝[copy]. 如果列表 list 是一个点列表[dotted list], 那么产生的列表[list]也是一个点列表[dotted list].
    
      只有列表 list 的列表结构[list struct]被拷贝; 产生的列表中的元素[element]和给定列表 list 中的对应元素[element]相同.
    
  • 示例(Examples):

    (setq lst (list 1 (list 2 3))) =>  (1 (2 3))
    (setq slst lst) =>  (1 (2 3))
    (setq clst (copy-list lst)) =>  (1 (2 3))
    (eq slst lst) =>  true
    (eq clst lst) =>  false
    (equal clst lst) =>  true
    (rplaca lst "one") =>  ("one" (2 3))
    slst =>  ("one" (2 3))
    clst =>  (1 (2 3))
    (setf (caadr lst) "two") =>  "two"
    lst =>  ("one" ("two" 3))
    slst =>  ("one" ("two" 3))
    clst =>  (1 ("two" 3))
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果列表 list 是一个环状列表[circular list]那么后果是未定义的.
    
  • 参见(See Also):

      copy-alist, copy-seq, copy-tree
    
  • 注意(Notes):

      创建的拷贝和列表 list 是 equal 的, 但不是 eq 的. 
    

函数 LIST, LIST*

  • 语法(Syntax):

      list &rest objects => list
    
      list* &rest objects+ => result
    
  • 参数和值(Arguments and Values):

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

      list 返回一个包含提供的那些对象 objects 的列表[list].
    
      list* 类似于 list, 除了给 list 的最后一个实参[argument]为构造的最后一个 cons 的 car, 而给 list* 的最后一个实参[argument]是构造的最后一个 cons 的 cdr. 因此, 任何给定的对 list* 的调用总是产生比相同数量参数的 list 调用少一个 cons.
    
      如果给 list* 的最后一个实参[argument]是一个列表[list], 那么效果就是构造一个新的相似的列表[list], 但是有着额外元素被添加到前面, 这些元素和 list* 的前面实参[argument]对应.
    
      如果 list* 只接收到一个对象 object, 那么返回那个对象 object, 不管它是否为一个列表[list].
    
  • 示例(Examples):

    (list 1) =>  (1)
    (list* 1) =>  1
    (setq a 1) =>  1
    (list a 2) =>  (1 2)
    '(a 2) =>  (A 2)
    (list 'a 2) =>  (A 2)
    (list* a 2) =>  (1 . 2)
    (list) =>  NIL ;i.e.,  ()
    (setq a '(1 2)) =>  (1 2)
    (eq a (list* a)) =>  true
    (list 3 4 'a (car '(b . c)) (+ 6 -2)) =>  (3 4 A B 4)
    (list* 'a 'b 'c 'd) ==  (cons 'a (cons 'b (cons 'c 'd))) =>  (A B C . D)
    (list* 'a 'b 'c '(d e f)) =>  (A B C D E F)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      cons
    
  • 注意(Notes):

      (list* x) ==  x
    

函数 LIST-LENGTH

  • 语法(Syntax):

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

      list---一个正规列表[proper list]或一个环状列表[circular list].
      length---一个非负整数[integer], 或者 nil.
    
  • 描述(Description):

      如果列表 list 是一个正规列表[proper list]就返回这个列表 list 的长度[length]. 如果列表 list 是一个环状列表[circular list]就返回 nil.
    
  • 示例(Examples):

    (list-length '(a b c d)) =>  4
    (list-length '(a (b c) d)) =>  3
    (list-length '()) =>  0
    (list-length nil) =>  0
    (defun circular-list (&rest elements)
      (let ((cycle (copy-list elements))) 
        (nconc cycle cycle)))
    (list-length (circular-list 'a 'b)) =>  NIL
    (list-length (circular-list 'a)) =>  NIL
    (list-length (circular-list)) =>  0
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 list 不是一个正规列表[proper list]或者一个环状列表[circular list], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      length
    
  • 注意(Notes):

      list-length 可以通过以下方式来实现:
    
    (defun list-length (x)  
      (do ((n 0 (+ n 2))           ;Counter.
            (fast x (cddr fast))    ;Fast pointer: leaps by 2.
            (slow x (cdr slow)))    ;Slow pointer: leaps by 1.
          (nil)
        ;; If fast pointer hits the end, return the count.
        (when (endp fast) (return n))
        (when (endp (cdr fast)) (return (+ n 1)))
        ;; If fast pointer eventually equals slow pointer,
        ;;  then we must be stuck in a circular list.
        ;; (A deeper property is the converse: if we are
        ;;  stuck in a circular list, then eventually the
        ;;  fast pointer will equal the slow pointer.
        ;;  That fact justifies this implementation.)
        (when (and (eq fast slow) (> n 0)) (return nil))))
    

函数 LISTP

  • 语法(Syntax):

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

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

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

    (listp nil) =>  true
    (listp (cons 1 2)) =>  true
    (listp (make-array 6)) =>  false
    (listp t) =>  false
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      consp
    
  • 注意(Notes):

      如果对象 object 是一个 cons, listp 不会检查对象 object 是否为一个正规列表[proper list]; 对于任何种类的列表[list]它都返回 true.
    
      (listp object) ==  (typep object 'list) ==  (typep object '(or cons null))
    

函数 MAKE-LIST

  • 语法(Syntax):

      make-list size &key initial-element => list
    
  • 参数和值(Arguments and Values):

      size---一个非负整数[integer].
      initial-element---一个对象[object]. 默认是 nil.
      list---一个列表[list].
    
  • 描述(Description):

      返回长度为给定大小 size 的列表[list], 其中每一个元素[element]都是初始化元素 initial-element.
    
  • 示例(Examples):

    (make-list 5) =>  (NIL NIL NIL NIL NIL)
    (make-list 3 :initial-element 'rah) =>  (RAH RAH RAH)
    (make-list 2 :initial-element '(1 2 3)) =>  ((1 2 3) (1 2 3))
    (make-list 0) =>  NIL ;i.e.,  ()
    (make-list 0 :initial-element 'new-element) =>  NIL 
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 size 不是一个非负整数[integer], 那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      cons, list
    
  • 注意(Notes): None.

宏 PUSH

  • 语法(Syntax):

      push item place => new-place-value
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      place---一个位置[place], 它的值[value]可能是任何对象[object].
      new-place-value---一个列表[list] (这个 place 的新值[value]).
    
  • 描述(Description):

      push 在存储在 place 的列表[list]前面加上 item, 把产生的列表[list]存储到 place, 然后返回那个列表[list].
    
      关于 place 的子表达式形式[subform]求值[evaluation]的信息, 见章节 5.1.1.1 (位置的子表达式形式求值).
    
  • 示例(Examples):

    (setq llst '(nil)) =>  (NIL)
    (push 1 (car llst)) =>  (1)
    llst =>  ((1))
    (push 1 (car llst)) =>  (1 1)
    llst =>  ((1 1))
    (setq x '(a (b c) d)) =>  (A (B C) D)
    (push 5 (cadr x)) =>  (5 B C)  
    x =>  (A (5 B C) D)
    
  • 副作用(Side Effects):

      place 的内容会被修改.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      pop, pushnew, 章节 5.1 (广义引用)
    
  • 注意(Notes):

      (push item place) 的效果等价于
    
      (setf place (cons item place))
    
      除了这个 place 的子表达式形式[subform]只被求值一次, 并且这个 item 在 place 之前被求值. 
    

宏 POP

  • 语法(Syntax):

      pop place => element
    
  • 参数和值(Arguments and Values):

      place---一个位置[place], 它的值[value]是一个列表[list] (可能, 但不是必须, 是一个点列表[dotted list]或环状列表[circular]).
      element---一个对象[object] (这个 place 的内容的 car).
    
  • 描述(Description):

      pop 读取[read] place 的值[value], 记住获取到的列表的 car, 把这个列表[list]的 cdr 写[write]回到那个 place 中, 并且最后产生[yield]这个原始获取的列表[list]的 car 部分.
    
      关于 place 的子表达式形式[subform]求值[evaluation]的信息, 见章节 5.1.1.1 (位置的子表达式形式求值).
    
  • 示例(Examples):

    (setq stack '(a b c)) =>  (A B C)
    (pop stack) =>  A  
    stack =>  (B C)
    (setq llst '((1 2 3 4))) =>  ((1 2 3 4))
    (pop (car llst)) =>  1
    llst =>  ((2 3 4))
    
  • 副作用(Side Effects):

      place 的内容会被修改.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      push, pushnew, 章节 5.1 (广义引用)
    
  • 注意(Notes):

      (pop place) 的效果大致等价于
    
      (prog1 (car place) (setf place (cdr place)))
    
      除了后面的可能会求值 place 中的任意子表达式形式[subform]三次, 而 pop 只求值它们一次. 
    

访问器 FIRST, SECOND, THIRD, FOURTH, FIFTH, SIXTH, SEVENTH, EIGHTH, NINTH, TENTH

  • 语法(Syntax):

      first list => object
    
      second list => object
    
      third list => object
    
      fourth list => object
    
      fifth list => object
    
      sixth list => object
    
      seventh list => object
    
      eighth list => object
    
      ninth list => object
    
      tenth list => object
    
      (setf (first list) new-object)
    
      (setf (second list) new-object)
    
      (setf (third list) new-object)
    
      (setf (fourth list) new-object)
    
      (setf (fifth list) new-object)
    
      (setf (sixth list) new-object)
    
      (setf (seventh list) new-object)
    
      (setf (eighth list) new-object)
    
      (setf (ninth list) new-object)
    
      (setf (tenth list) new-object)
    
  • 参数和值(Arguments and Values):

      list---一个列表[list], 它可能是一个点列表[dotted list]或者一个环状列表[circular list].
      object, new-object---一个对象[object].
    
  • 描述(Description):

      函数 first, second, third, fourth, fifth, sixth, seventh, eighth, ninth, 和 tenth 分别访问列表 list 的第一, 第二, 第三, 第四, 第五, 第六, 第七, 第八, 第九, 和第十的元素[element]. 具体地说,
    
      (first list)    ==   (car list)
      (second list)   ==   (car (cdr list))
      (third list)    ==   (car (cddr list))
      (fourth list)   ==   (car (cdddr list))
      (fifth list)    ==   (car (cddddr list))
      (sixth list)    ==   (car (cdr (cddddr list)))
      (seventh list)  ==   (car (cddr (cddddr list)))
      (eighth list)   ==   (car (cdddr (cddddr list)))
      (ninth list)    ==   (car (cddddr (cddddr list)))
      (tenth list)    ==   (car (cdr (cddddr (cddddr list))))
    
      setf 也可以和这些函数中的任何一个一起使用来改变一个已存在的成分. 应用相同的等价性. 比如:
    
      (setf (fifth list) new-object) ==  (setf (car (cddddr list)) new-object)
    
  • 示例(Examples):

    (setq lst '(1 2 3 (4 5 6) ((V)) vi 7 8 9 10)) 
    =>  (1 2 3 (4 5 6) ((V)) VI 7 8 9 10)
    (first lst) =>  1
    (tenth lst) =>  10
    (fifth lst) =>  ((V))
    (second (fourth lst)) =>  5
    (sixth '(1 2 3)) =>  NIL
    (setf (fourth lst) "four") =>  "four"
    lst =>  (1 2 3 "four" ((V)) VI 7 8 9 10)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      car, nth
    
  • 注意(Notes):

      first 功能上等价于 car, second 功能上等价于 cadr, third 功能上等价于 caddr, 并且 fourth 功能上等价于 cadddr.
    
      这里使用的序号是从一开始的, 相比于 nth 使用的从零开始的序号:
    
      (fifth x) ==  (nth 4 x)
    

访问器 NTH

  • 语法(Syntax):

      nth n list => object
    
      (setf (nth n list) new-object)
    
  • 参数和值(Arguments and Values):

      n---一个非负整数[integer].
      list---一个列表[list], 它可能是一个点列表[dotted list]或一个环状列表[circular list].
      object---一个对象[object].
      new-object---一个对象[object].
    
  • 描述(Description):

      nth 查找列表 list 的第 n 个元素, 其中这个列表 list 的 car 是第 0 个元素. 具体来说,
    
      (nth n list) ==  (car (nthcdr n list))
    
      nth 可以被用于给 setf 指定一个 place. 具体来说,
    
      (setf (nth n list) new-object) ==  (setf (car (nthcdr n list)) new-object)
    
  • 示例(Examples):

    (nth 0 '(foo bar baz)) =>  FOO
    (nth 1 '(foo bar baz)) =>  BAR
    (nth 3 '(foo bar baz)) =>  NIL
    (setq 0-to-3 (list 0 1 2 3)) =>  (0 1 2 3)
    (setf (nth 2 0-to-3) "two") =>  "two"
    0-to-3 =>  (0 1 "two" 3)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      elt, first, nthcdr
    
  • 注意(Notes): None.

函数 ENDP

  • 语法(Syntax):

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

      list---一个列表[list], 它可能是一个点列表[dotted list]或一个环状列表[circular list].
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      如果列表 list 是一个空列表[empty list]就返回 true. 如果列表 list 是一个 cons 就返回 false.
    
  • 示例(Examples):

    (endp nil) =>  true
    (endp '(1 2)) =>  false
    (endp (cddr '(1 2))) =>  true
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 list 不是一个列表就应该发出一个 type-error 类型的错误.
    
  • 参见(See Also): None.

  • 注意(Notes):

      endp 的目的是测试正规列表的结束. 由于 endp 不会进入到一个 cons 中, 所以明确定义了传递给它点列表[dotted list]的情况. 然而, 如果通过在一个点列表[dotted list]上反复调用 cdr 来产生更短的"列表"并且使用 endp 来测试这些"列表", 当这个非 nil [non-nil]的原子[atom] (它事实上不是一个列表[list]) 最终称为给 endp 的参数时, 最后会产生一个有着不确定后果的情况. 由于这是使用 endp 的常用方式, 所以它是保守的编程风格, 并且与 endp 的意图一致, 即把 endp 当作一个在正规列表[proper list]上的函数, 而不是强制一个正规列表[proper list]的参数类型, 除非参数是原子的[atomic]. 
    

函数 NULL

  • 语法(Syntax):

      null object => boolean
    
  • 参数和值(Arguments and Values):

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

      如果对象 object 是一个空列表[empty list]就返回 t; 否则, 返回 nil.
    
  • 示例(Examples):

    (null '()) =>  T
    (null nil) =>  T
    (null t) =>  NIL
    (null 1) =>  NIL
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      not
    
  • 注意(Notes):

      null 意图在于测试空列表[empty list]而 not 意图在于反转一个 boolean (或广义的 boolean [generalized boolean]). 操作上, null 和 not 计算出相同的结果; 使用哪一个是风格的问题.
    
      (null object) ==  (typep object 'null) ==  (eq object '())
    

函数 NCONC

  • 语法(Syntax):

      nconc &rest lists => concatenated-list
    
  • 参数和值(Arguments and Values):

      list---除了最后一个的每个都必须是一个列表[list] (它可能是一个点列表[dotted list]但一定不是一个环状列表[circular list]); 最后的 list 可以是任何对象[object].
      concatenated-list---一个列表[list].
    
  • 描述(Description):

      返回这些列表 lists 的一个串接列表[list]. 如果没有提供列表 lists, (nconc) 返回 nil. nconc 使用以下递归关系来定义:
    
      (nconc) =>  ()
      (nconc nil . lists) ==  (nconc . lists)
      (nconc list) =>  list
      (nconc list-1 list-2) ==  (progn (rplacd (last list-1) list-2) list-1)
      (nconc list-1 list-2 . lists) ==  (nconc (nconc list-1 list-2) . lists)
    
  • 示例(Examples):

    (nconc) =>  NIL
    (setq x '(a b c)) =>  (A B C)
    (setq y '(d e f)) =>  (D E F)
    (nconc x y) =>  (A B C D E F)
    x =>  (A B C D E F)
    
      注意, 在这个例子中, 现在那个 x 的值是不同的, 因为它的最后一个 cons 已经被 rplacd 为 y 的值. 如果 (nconc x y) 被再一次求值, 它会产生一个环状列表[circular list], 它的打印表示会是 (A B C D E F D E F D E F ...), 一直重复; 如果这个 *print-circle* 开关不是 nil, 它会被打印为 (A B C . #1=(D E F . #1#)).
    
    (setq foo (list 'a 'b 'c 'd 'e)
          bar (list 'f 'g 'h 'i 'j)
          baz (list 'k 'l 'm)) =>  (K L M)
    (setq foo (nconc foo bar baz)) =>  (A B C D E F G H I J K L M)
    foo =>  (A B C D E F G H I J K L M)
    bar =>  (F G H I J K L M)
    baz =>  (K L M)
    
    (setq foo (list 'a 'b 'c 'd 'e)
          bar (list 'f 'g 'h 'i 'j)
          baz (list 'k 'l 'm)) =>  (K L M)
    (setq foo (nconc nil foo bar nil baz)) =>  (A B C D E F G H I J K L M) 
    foo =>  (A B C D E F G H I J K L M)
    bar =>  (F G H I J K L M)
    baz =>  (K L M)
    
  • 副作用(Side Effects):

      这些列表 lists 会被修改而不是拷贝.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      append, concatenate
    
  • 注意(Notes): None.

函数 APPEND

  • 语法(Syntax):

      append &rest lists => result
    
  • 参数和值(Arguments and Values):

      list---除了最后一个以外的每一个都必须是一个正规列表[proper list], 最后一个可能是任何对象[object].
      result---一个对象[object]. 这个会是一个列表[list], 除非最后一个 list 不是一个列表[list]并且所有前面的 lists 都是 null.
    
  • 描述(Description):

      append 返回一个由那些列表 lists 的拷贝拼接而成的一个新列表. 那些列表 lists 保持不变; 这些列表 lists 中除了最后一个以外的每一个的列表结构[list struct]都会被拷贝. 最后一个参数不会被拷贝; 它成为前面那些列表 lists 拼接后的最后一个点对[dotted pair]的 cdr, 如果前面没有非空[non-empty]列表就直接返回.
    
  • 示例(Examples):

    (append '(a b c) '(d e f) '() '(g)) =>  (A B C D E F G)
    (append '(a b c) 'd) =>  (A B C . D)
    (setq lst '(a b c)) =>  (A B C)
    (append lst '(d)) =>  (A B C D)
    lst =>  (A B C)
    (append) =>  NIL
    (append 'a) =>  A
    ```LISP
    
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      nconc, concatenate
    
  • 注意(Notes): None.

函数 REVAPPEND, NRECONC

  • 语法(Syntax):

      revappend list tail => result-list
    
      nreconc list tail => result-list
    
  • 参数和值(Arguments and Values):

      list---一个正规列表[proper list].
      tail---一个对象[object].
      result-list---一个对象[object].
    
  • 描述(Description):

      revappend 构造一个列表 list 的拷贝[copy[2]], 但是其中的元素[element]是以相反的顺序. 它接下来追加 (就像是通过 nconc) 那个 tail 到那个反转后的列表并返回那个结果.
    
      nreconc 反转在列表 list 中元素[element]的顺序 (就像是通过 nreverse). 它接下来追加 (就像是通过 nconc) 那个 tail 给那个反转后的列表并返回那个结果.
    
      产生的列表[list]和 tail 共享列表结构[list structure].
    
  • 示例(Examples):

    (let ((list-1 (list 1 2 3))
          (list-2 (list 'a 'b 'c)))
      (print (revappend list-1 list-2))
      (print (equal list-1 '(1 2 3)))
      (print (equal list-2 '(a b c))))
    >>  (3 2 1 A B C) 
    >>  T
    >>  T
    =>  T
    
    (revappend '(1 2 3) '()) =>  (3 2 1)
    (revappend '(1 2 3) '(a . b)) =>  (3 2 1 A . B)
    (revappend '() '(a b c)) =>  (A B C)
    (revappend '(1 2 3) 'a) =>  (3 2 1 . A)
    (revappend '() 'a) =>  A   ;degenerate case
    
    (let ((list-1 '(1 2 3))
          (list-2 '(a b c)))
      (print (nreconc list-1 list-2))
      (print (equal list-1 '(1 2 3)))
      (print (equal list-2 '(a b c))))
    >>  (3 2 1 A B C) 
    >>  NIL
    >>  T
    =>  T
    
  • 副作用(Side Effects):

      revappend 不会修改它的任何一个实参[argument]. nreconc 允许去修改列表 list 但是不能修改 tail.
    
      虽然它可能被不同地实现, nreconc 被限制为有着和下面这个等价的副作用行为:
    
      (nconc (nreverse list) tail)
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      reverse, nreverse, nconc
    
  • 注意(Notes):

      下面这些功能等价性是对的, 虽然好的实现[implementation]通常使用一个更快的算法来取得相同的效果:
    
      (revappend list tail) ==  (nconc (reverse list) tail)
      (nreconc list tail) ==  (nconc (nreverse list) tail)
    

函数 BUTLAST, NBUTLAST

  • 语法(Syntax):

      butlast list &optional n => result-list
    
      nbutlast list &optional n => result-list
    
  • 参数和值(Arguments and Values):

      list---一个列表[list], 它可能是一个点列表[dotted list]但一定不是一个环状列表[circular list].
      n---一个非负整数[integer].
      result-list---一个列表[list].
    
  • 描述(Description):

      butlast 返回列表 list 的一个拷贝, 其中最后 n 个 cons 会被省略. 如果 n 没有被提供, 它的值就是 1. 如果在列表 list 中的 cons 少于 n 个, 返回 nil, 并且在 nbutlast 的情况中, 列表 list 不会被修改.
    
      nbutlast 类似于 butlast, 但是 nbutlast 可以修改列表 list. 它把这个列表 list 从结尾开始的 n+1 个 cons 的 cdr 改变为 nil.
    
  • 示例(Examples):

    (setq lst '(1 2 3 4 5 6 7 8 9)) =>  (1 2 3 4 5 6 7 8 9)
    (butlast lst) =>  (1 2 3 4 5 6 7 8)
    (butlast lst 5) =>  (1 2 3 4)
    (butlast lst (+ 5 5)) =>  NIL
    lst =>  (1 2 3 4 5 6 7 8 9)
    (nbutlast lst 3) =>  (1 2 3 4 5 6)
    lst =>  (1 2 3 4 5 6)
    (nbutlast lst 99) =>  NIL
    lst =>  (1 2 3 4 5 6)
    (butlast '(a b c d)) =>  (A B C)
    (butlast '((a b) (c d))) =>  ((A B))
    (butlast '(a)) =>  NIL
    (butlast nil) =>  NIL
    (setq foo (list 'a 'b 'c 'd)) =>  (A B C D)
    (nbutlast foo) =>  (A B C)
    foo =>  (A B C)
    (nbutlast (list 'a)) =>  NIL
    (nbutlast '()) =>  NIL
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list 不是一个正规列表[proper list]或一个点列表[dotted list]那么应该发出一个 type-error 类型[type]的错误. 如果 n 不是一个非负整数[integer]那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also): None.

  • 注意(Notes):

      (butlast list n) ==  (ldiff list (last list n))
    

函数 LAST

  • 语法(Syntax):

      last list &optional n => tail
    
  • 参数和值(Arguments and Values):

      list---一个列表[list], 它可能是一个点列表[dotted list]但一定不是一个环状列表[circular list].
      n---一个非负整数[integer]. 默认是 1.
      tail---一个对象[object].
    
  • 描述(Description):

      last 返回列表 list 的最后 n 个 cons (不是最后 n 个元素). 如果列表 list 是 (), last 返回 ().
    
      如果 n 是 0, 那么终止这个列表 list 的原子会被返回. 如果 n 大于或等于列表 list 中的 cons 存储格的数量, 那么结果就是那个列表 list.
    
  • 示例(Examples):

    (last nil) =>  NIL
    (last '(1 2 3)) =>  (3)
    (last '(1 2 . 3)) =>  (2 . 3)
    (setq x (list 'a 'b 'c 'd)) =>  (A B C D)
    (last x) =>  (D)
    (rplacd (last x) (list 'e 'f)) x =>  (A B C D E F)
    (last x) =>  (F)
    
    (last '(a b c))   =>  (C)
    
    (last '(a b c) 0) =>  ()
    (last '(a b c) 1) =>  (C)
    (last '(a b c) 2) =>  (B C)
    (last '(a b c) 3) =>  (A B C)
    (last '(a b c) 4) =>  (A B C)
    
    (last '(a . b) 0) =>  B
    (last '(a . b) 1) =>  (A . B)
    (last '(a . b) 2) =>  (A . B)
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果列表 list 是一个环状列表[circular list]那么后果是未定义的. 如果 n 不是一个非负整数[integer]那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      butlast, nth
    
  • 注意(Notes):

      以下代码可以被用于定义 last.
    
      (defun last (list &optional (n 1))
        (check-type n (integer 0))
        (do ((l list (cdr l))
              (r list)
              (i 0 (+ i 1)))
            ((atom l) r)
          (if (>= i n) (pop r))))
    

函数 LDIFF, TAILP

  • 语法(Syntax):

      ldiff list object => result-list
    
      tailp object list => generalized-boolean
    
  • 参数和值(Arguments and Values):

      list---一个列表[list], 它可能是一个点列表[dotted list].
      object---一个对象[object].
      result-list---一个列表[list].
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      如果对象 object 和 list 的末端[tail]相同[same], tailp 返回 true; 否则, 它返回 false.
    
      如果对象 object 和 list 的末端[tail]相同[same], ldiff 返回一个包含列表 list 的列表结构[list structure]中在 object 之前的列表[list]元素[element]的新的[fresh]列表[list]; 否则, 它返回列表 list 的一个拷贝[copy[2]].
    
  • 示例(Examples):

    (let ((lists '#((a b c) (a b c . d))))
      (dotimes (i (length lists)) ()
        (let ((list (aref lists i)))
          (format t "~2&list=~S ~21T(tailp object list)~
                      ~44T(ldiff list object)~%" list)
            (let ((objects (vector list (cddr list) (copy-list (cddr list))
                                    '(f g h) '() 'd 'x)))
              (dotimes (j (length objects)) ()
                (let ((object (aref objects j)))
                  (format t "~& object=~S ~21T~S ~44T~S"
                          object (tailp object list) (ldiff list object))))))))
    >>  
    >>  list=(A B C)         (tailp object list)    (ldiff list object)
    >>   object=(A B C)      T                      NIL
    >>   object=(C)          T                      (A B)
    >>   object=(C)          NIL                    (A B C)
    >>   object=(F G H)      NIL                    (A B C)
    >>   object=NIL          T                      (A B C)
    >>   object=D            NIL                    (A B C)
    >>   object=X            NIL                    (A B C)
    >>  
    >>  list=(A B C . D)     (tailp object list)    (ldiff list object)
    >>   object=(A B C . D)  T                      NIL
    >>   object=(C . D)      T                      (A B)
    >>   object=(C . D)      NIL                    (A B C . D)
    >>   object=(F G H)      NIL                    (A B C . D)
    >>   object=NIL          NIL                    (A B C . D)
    >>   object=D            T                      (A B C)
    >>   object=X            NIL                    (A B C . D)
    =>  NIL
    ```LISP
    
    
  • 副作用(Side Effects):

      不管是 ldiff 函数 tailp 都不会修改它的任何一个实参[argument].
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list 不是一个正规列表[proper list]或者一个点列表[dotted list]那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      set-difference
    
  • 注意(Notes):

      如果列表 list 是一个环状列表[circular list], 当且仅当给定对象 object 是 list 的一个末端[tail]时 tailp 会确实地产生[yield]一个值[value]. 否则, 后果是未指定的: 一个检测环状的给定实现[implementation]必须返回 false, 但是由于一个实现[implementation]没有义务去检测这样一种情况[situation], 所以 tailp 可能会在不返回的情况下无限循环.
    
      tailp 可以通过如下来定义:
    
    (defun tailp (object list)
      (do ((list list (cdr list)))
          ((atom list) (eql list object))
          (if (eql object list)
              (return t))))
    
      并且 ldiff 可以通过如下来定义:
    
    (defun ldiff (list object)
      (do ((list list (cdr list))
          (r '() (cons (car list) r)))
          ((atom list)
          (if (eql list object) (nreverse r) (nreconc r list)))
        (when (eql object list)
          (return (nreverse r)))))
    

函数 NTHCDR

  • 语法(Syntax):

      nthcdr n list => tail
    
  • 参数和值(Arguments and Values):

      n---一个非负整数[integer].
      list---一个列表[list], 它可能是一个点列表[dotted list]或一个环状列表[circular list].
      tail---一个对象[object].
    
  • 描述(Description):

      返回通过成功调用 n 次 cdr 得到的列表 list 的末端[tail].
    
  • 示例(Examples):

    (nthcdr 0 '()) =>  NIL
    (nthcdr 3 '()) =>  NIL
    (nthcdr 0 '(a b c)) =>  (A B C)
    (nthcdr 2 '(a b c)) =>  (C)
    (nthcdr 4 '(a b c)) =>  ()
    (nthcdr 1 '(0 . 1)) =>  1
    
    (locally (declare (optimize (safety 3)))
      (nthcdr 3 '(0 . 1)))
    Error: Attempted to take CDR of 1.
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 n 不是一个非负整数[integer]那么应该发出一个 type-error 类型[type]的错误.
    
      对于大于 1 的整数 n, 通过 (nthcdr n list) 完成的错误检测和 (nthcdr (- n 1) (cdr list)) 相同; 见函数[function] cdr.
    
  • 参见(See Also):

      cdr, nth, rest
    
  • 注意(Notes): None.

访问器 REST

  • 语法(Syntax):

      rest list => tail
    
      (setf (rest list) new-tail)
    
  • 参数和值(Arguments and Values):

      list---一个列表[list], 它可能是一个点列表[dotted list]或一个环状列表[circular list].
      tail---一个对象[object].
    
  • 描述(Description):

      rest 执行和 cdr 相同的操作, 但是记忆上与 first 互补. 具体来说,
    
      (rest list) ==  (cdr list)
      (setf (rest list) new-tail) ==  (setf (cdr list) new-tail)
    
  • 示例(Examples):

    (rest '(1 2)) =>  (2)
    (rest '(1 . 2)) =>  2
    (rest '(1)) =>  NIL
    (setq *cons* '(1 . 2)) =>  (1 . 2)
    (setf (rest *cons*) "two") =>  "two"
    *cons* =>  (1 . "two")
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      cdr, nthcdr
    
  • 注意(Notes):

      当这个参数被主观上视作一个列表[list]而不是一个 cons 时, rest 通常在风格上优于 cdr. 
    

函数 MEMBER, MEMBER-IF, MEMBER-IF-NOT

  • 语法(Syntax):

      member item list &key key test test-not => tail
    
      member-if predicate list &key key => tail
    
      member-if-not predicate list &key key => tail
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      list---一个正规列表[proper list].
      predicate---返回一个广义 boolean [generalized boolean]的单实参[argument]函数[function]的标识符[designator].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单实参[argument]函数[function]的标识符[designator], 或者 nil.
      tail---一个列表[list].
    
  • 描述(Description):

      member, member-if, 和 member-if-not 每一个都在列表 list 中搜索 item 或者一个满足测试条件[satisfy the test]的顶层元素. 给这个断言函数 predicate 的参数是列表 list 的一个元素.
    
      如果某个元素满足测试条件[satisfy the test], 这个列表 list 从这个元素开始到末端的尾部会被返回; 否则返回 nil.
    
      只在 list 的顶层搜索.
    
  • 示例(Examples):

    (member 2 '(1 2 3)) =>  (2 3)                                 
    (member 2 '((1 . 2) (3 . 4)) :test-not #'= :key #'cdr) =>  ((3 . 4))
    (member 'e '(a b c d)) =>  NIL
    
    (member-if #'listp '(a b nil c d)) =>  (NIL C D)
    (member-if #'numberp '(a #\Space 5/3 foo)) =>  (5/3 FOO)
    (member-if-not #'zerop 
                    '(3 6 9 11 . 12)
                    :key #'(lambda (x) (mod x 3))) =>  (11 . 12)
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 list 不是一个正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      find, position, 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数被废弃了.
    
      函数[function] member-if-not 被废弃了.
    
      在下面这个中
    
      (member 'a '(g (a y) c a d e a f)) =>  (A D E A F)
    
      member 返回的值和列表[list]中以 a 开始的部分相同[identical]. 因此在 member 的结果上调用 rplaca 可以被用来修改列表[list]中 a 被找到的部分 (假定已经检测了这个 member 没有返回 nil). 
    

函数 MAPC, MAPCAR, MAPCAN, MAPL, MAPLIST, MAPCON

  • 语法(Syntax):

      mapc function &rest lists+ => list-1
    
      mapcar function &rest lists+ => result-list
    
      mapcan function &rest lists+ => concatenated-results
    
      mapl function &rest lists+ => list-1
    
      maplist function &rest lists+ => result-list
    
      mapcon function &rest lists+ => concatenated-results
    
  • 参数和值(Arguments and Values):

      function---接受和列表 lists 数量相同的实参[argument]的函数[function]标识符[designator].
      list---一个正规列表[proper list].
      list-1---第一个列表 list (它一定是一个正规列表[proper list]).
      result-list---一个列表[list].
      concatenated-results---一个列表[list].
    
  • 描述(Description):

      映射操作应用函数 function 到参数的连续集合上, 其中一个参数是从每一个序列[sequence]获取而来. 除了 mapc 和 mapl, 结果都包含了函数 function 返回的结果. 在 mapc 和 mapl 的情况下, 产生的序列[sequence]是列表 list.
    
      函数 function 首先在所有索引为 0 的元素上调用, 然后在所有索引为 1 的元素上调用, 以此类推. result-type 指定了产生序列[sequence]的类型[type]. 如果函数 function 是一个符号[symbol], 它被强制转为[coerce]一个函数[function], 就像是通过 symbol-function 一样.
    
      mapcar 在那些列表 lists 的连续元素[element]上操作. 函数 function 被应用于每个列表 list 的第一个元素[element]上, 然后到每个列表 list 的第二个元素[element]上, 以此类推. 这个迭代操作在最短的列表耗尽时终止, 而在其他列表中的过量元素会被忽略. 由 mapcar 返回的值是对函数 function 连续调用的结果组成的列表.
    
      mapc 类似于 mapcar 除了应用函数 function 的结果不会被累积. 这个 list 参数会被返回.
    
      maplist 类似于 mapcar 除了函数 function 被应用于列表 lists 的连续子列表中. 函数 function 首先被应用到这些 lists 自身, 然后被应用到每个列表 list 的 cdr, 然后是每个列表的 cdr 的 cdr, 以此类推.
    
      mapl 类似于 maplist 除了应用函数 function 的结果不会被累积; list-1 会被返回.
    
      mapcan 和 mapcon 分别类似于 mapcar 和 maplist, 除了应用函数 function 的结果会通过使用 nconc 而不是 list 被组合到一个列表[list]中. 这也就是说,
    
      (mapcon f x1 ... xn)
        ==  (apply #'nconc (maplist f x1 ... xn))
    
      并且 mapcan 和 mapcar 之间的关系类似.
    
  • 示例(Examples):

    (mapcar #'car '((1 a) (2 b) (3 c))) =>  (1 2 3) 
    (mapcar #'abs '(3 -4 2 -5 -6)) =>  (3 4 2 5 6)
    (mapcar #'cons '(a b c) '(1 2 3)) =>  ((A . 1) (B . 2) (C . 3))
    
    (maplist #'append '(1 2 3 4) '(1 2) '(1 2 3)) 
    =>  ((1 2 3 4 1 2 1 2 3) (2 3 4 2 2 3)) 
    (maplist #'(lambda (x) (cons 'foo x)) '(a b c d))
    =>  ((FOO A B C D) (FOO B C D) (FOO C D) (FOO D))
    (maplist #'(lambda (x) (if (member (car x) (cdr x)) 0 1)) '(a b a c d b c))
    =>  (0 0 1 0 1 1 1)
    ;An entry is 1 if the corresponding element of the input
    ;  list was the last instance of that element in the input list.
    
    (setq dummy nil) =>  NIL 
    (mapc #'(lambda (&rest x) (setq dummy (append dummy x)))
            '(1 2 3 4)
            '(a b c d e)
            '(x y z)) =>  (1 2 3 4) 
    dummy =>  (1 A X 2 B Y 3 C Z)                   
    
    (setq dummy nil) =>  NIL 
    (mapl #'(lambda (x) (push x dummy)) '(1 2 3 4)) =>  (1 2 3 4) 
    dummy =>  ((4) (3 4) (2 3 4) (1 2 3 4)) 
    
    (mapcan #'(lambda (x y) (if (null x) nil (list x y)))
              '(nil nil nil d e)
              '(1 2 3 4 5 6)) =>  (D 4 E 5) 
    (mapcan #'(lambda (x) (and (numberp x) (list x)))
              '(a 1 b c 3 4 d 5))
    =>  (1 3 4 5)
    
      在这个情况中函数被当作一个过滤器; 这是使用 mapcan 的标准 Lisp 风格.
    
    (mapcon #'list '(1 2 3 4)) =>  ((1 2 3 4) (2 3 4) (3 4) (4)) 
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果任何 list 不是一个正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

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

函数 ACONS

  • 语法(Syntax):

      acons key datum alist => new-alist
    
  • 参数和值(Arguments and Values):

      key---一个对象[object].
      datum---一个对象[object].
      alist---一个关联列表[association list].
      new-alist---一个关联列表[association list].
    
  • 描述(Description):

      创建一个新的[fresh] cons, 其中的 cdr 是 alist 而 car 是另一个新的[fresh] cons, 这个新的 cons 的 car 是键 key 而 cdr 是对象 datum.
    
  • 示例(Examples):

    (setq alist '()) =>  NIL
    (acons 1 "one" alist) =>  ((1 . "one"))
    alist =>  NIL
    (setq alist (acons 1 "one" (acons 2 "two" alist))) =>  ((1 . "one") (2 . "two"))
    (assoc 1 alist) =>  (1 . "one")
    (setq alist (acons 1 "uno" alist)) =>  ((1 . "uno") (1 . "one") (2 . "two"))
    (assoc 1 alist) =>  (1 . "uno")
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      assoc, pairlis
    
  • 注意(Notes):

      (acons key datum alist) ==  (cons (cons key datum) alist)
    

函数 ASSOC, ASSOC-IF, ASSOC-IF-NOT

  • 语法(Syntax):

      assoc item alist &key key test test-not => entry
    
      assoc-if predicate alist &key key => entry
    
      assoc-if-not predicate alist &key key => entry
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      alist---一个关联列表[association list].
      predicate---返回一个广义 boolean [generalized boolean]的单实参[argument]函数[function]的标识符[designator].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---一个返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      entry---一个 cons, 是 alist 的一个元素[element], 或者是 nil.
    
  • 描述(Description):

      assoc, assoc-if, 和 assoc-if-not 返回在 alist 中的第一个 car 满足测试条件[satisfy the test]的 cons, 如果没有找到这样的 cons 就是 nil.
    
      对于 assoc, assoc-if, 和 assoc-if-not, 如果在 alist 中出现一个 nil 代替一个对, 那么它会被忽略.
    
  • 示例(Examples):

    (setq values '((x . 100) (y . 200) (z . 50))) =>  ((X . 100) (Y . 200) (Z . 50))
    (assoc 'y values) =>  (Y . 200)
    (rplacd (assoc 'y values) 201) =>  (Y . 201)
    (assoc 'y values) =>  (Y . 201)
    (setq alist '((1 . "one")(2 . "two")(3 . "three"))) 
    =>  ((1 . "one") (2 . "two") (3 . "three"))
    (assoc 2 alist) =>  (2 . "two")
    (assoc-if #'evenp alist) =>  (2 . "two")
    (assoc-if-not #'(lambda(x) (< x 3)) alist) =>  (3 . "three")
    (setq alist '(("one" . 1)("two" . 2))) =>  (("one" . 1) ("two" . 2))
    (assoc "one" alist) =>  NIL
    (assoc "one" alist :test #'equalp) =>  ("one" . 1)
    (assoc "two" alist :key #'(lambda(x) (char x 2))) =>  NIL 
    (assoc #\o alist :key #'(lambda(x) (char x 2))) =>  ("two" . 2)
    (assoc 'r '((a . b) (c . d) (r . x) (s . y) (r . z))) =>   (R . X)
    (assoc 'goo '((foo . bar) (zoo . goo))) =>  NIL
    (assoc '2 '((1 a b c) (2 b c d) (-7 x y z))) =>  (2 B C D)
    (setq alist '(("one" . 1) ("2" . 2) ("three" . 3)))
    =>  (("one" . 1) ("2" . 2) ("three" . 3))
    (assoc-if-not #'alpha-char-p alist
                  :key #'(lambda (x) (char x 0))) =>  ("2" . 2)
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 alist 不是一个关联列表[association list]那么应该准备发出一个 type-error 类型[type]的错误..
    
  • 参见(See Also):

      rassoc, find, member, position, 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数已经被废弃.
    
      函数[function] assoc-if-not 已经被废弃.
    
      有可能为了 "更新" alist 去 rplacd 这个 assoc 的结果, 假定它不是 nil.
    
      这两个表达式
    
      (assoc item list :test fn)
    
      和
    
      (find item list :test fn :key #'car)
    
      在意义上等价, 但是有一个例外: 如果 nil 出现在 alist 中替代一个对, 并且 item 是 nil, find 会计算 alist 中的 nil 的 car, 发现它等价于 item, 然后返回 nil, 反之 assoc 会忽略 alist 中的这个 nil 然后继续搜索一个实际的 car 为 nil 的 cons. 
    

函数 COPY-ALIST

  • 语法(Syntax):

      copy-alist alist => new-alist
    
  • 参数和值(Arguments and Values):

      alist---一个关联列表[association list].
      new-alist---一个关联列表[association list].
    
  • 描述(Description):

      copy-alist 返回 alist 的一个拷贝[copy].
    
      这个 alist 的列表结构[list structure]会被拷贝, 而 alist 中的 cons 元素[element]也会被拷贝 (仅仅只是作为 cons). 任何其他被 alist 引用的对象[object], 不管是直接还是间接的, 都会被继续共享.
    
  • 示例(Examples):

    (defparameter *alist* (acons 1 "one" (acons 2 "two" '())))
    *alist* =>  ((1 . "one") (2 . "two"))
    (defparameter *list-copy* (copy-list *alist*))
    *list-copy* =>  ((1 . "one") (2 . "two"))
    (defparameter *alist-copy* (copy-alist *alist*))
    *alist-copy* =>  ((1 . "one") (2 . "two"))
    (setf (cdr (assoc 2 *alist-copy*)) "deux") =>  "deux"
    *alist-copy* =>  ((1 . "one") (2 . "deux"))
    *alist* =>  ((1 . "one") (2 . "two"))
    (setf (cdr (assoc 1 *list-copy*)) "uno") =>  "uno"
    *list-copy* =>  ((1 . "uno") (2 . "two"))
    *alist* =>  ((1 . "uno") (2 . "two"))
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      copy-list
    
  • 注意(Notes): None.

函数 PAIRLIS

  • 语法(Syntax):

      pairlis keys data &optional alist => new-alist
    
  • 参数和值(Arguments and Values):

      keys---一个正规列表[proper list].
      data---一个正规列表[proper list].
      alist---一个关联列表[association list]. 默认是空列表[empty list].
      new-alist---一个关联列表[association list].
    
  • 描述(Description):

      返回一个关联 keys 的元素到对应 data 中的元素的关联列表[association list]. 如果 keys 和 data 不是相同长度[length]那么后果是未定义的.
    
      如果提供了 alist, pairlis 返回一个修改后的 alist, 这些新的对被添加到它的前面. 这些新的对可能以向前或向后的顺序出现在产生的关联列表[association list]中. 下面这个的结果
    
      (pairlis '(one two) '(1 2) '((three . 3) (four . 19)))
    
      可能是
    
      ((one . 1) (two . 2) (three . 3) (four . 19))
    
      或
    
      ((two . 2) (one . 1) (three . 3) (four . 19))
    
  • 示例(Examples):

    (setq keys '(1 2 3)
            data '("one" "two" "three")
            alist '((4 . "four"))) =>  ((4 . "four"))
    (pairlis keys data) =>  ((3 . "three") (2 . "two") (1 . "one"))
    (pairlis keys data alist)
    =>  ((3 . "three") (2 . "two") (1 . "one") (4 . "four"))
    alist =>  ((4 . "four"))
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果键 keys 和 data 不是正规列表[proper list]那么应该发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      acons
    
  • 注意(Notes): None.

函数 RASSOC, RASSOC-IF, RASSOC-IF-NOT

  • 语法(Syntax):

      rassoc item alist &key key test test-not => entry
    
      rassoc-if predicate alist &key key => entry
    
      rassoc-if-not predicate alist &key key => entry
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      alist---一个关联列表[association list].
      predicate---返回一个广义 boolean [generalized boolean]的单实参[argument]函数[function]的标识符[designator].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单实参[argument]函数[function]的标识符[designator], 或者 nil.
      entry---是 alist 中的一个 cons 元素[element], 或者是 nil.
    
  • 描述(Description):

      rassoc, rassoc-if, 和 rassoc-if-not 返回 cdr 满足测试条件[satisfy the test]的第一个 cons. 如果没有找到这样的 cons, 就返回 nil.
    
      如果 nil 出现在 alist 中替代了一个对, 它会被忽略.
    
  • 示例(Examples):

    (setq alist '((1 . "one") (2 . "two") (3 . 3))) 
    =>  ((1 . "one") (2 . "two") (3 . 3))
    (rassoc 3 alist) =>  (3 . 3)
    (rassoc "two" alist) =>  NIL
    (rassoc "two" alist :test 'equal) =>  (2 . "two")
    (rassoc 1 alist :key #'(lambda (x) (if (numberp x) (/ x 3)))) =>  (3 . 3)
    (rassoc 'a '((a . b) (b . c) (c . a) (z . a))) =>  (C . A)
    (rassoc-if #'stringp alist) =>  (1 . "one")
    (rassoc-if-not #'vectorp alist) =>  (3 . 3)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

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

      参数 :test-not 已经被废弃.
    
      函数[function] rassoc-if-not 已经被废弃.
    
      有可能为了 "更新" alist 去 rplacd 这个 rassoc 的结果, 假定它不是 nil.
    
      表达式
    
      (rassoc item list :test fn)
    
      和
    
      (find item list :test fn :key #'cdr)
    
      在意义上是等价的, 除了当这个 item 是 nil 并且 nil 出现在 alist 中替换一个对时. 见函数[function] assoc. 
    

函数 GET-PROPERTIES

  • 语法(Syntax):

      get-properties plist indicator-list => indicator, value, tail
    
  • 参数和值(Arguments and Values):

      plist---一个属性列表[property list].
      indicator-list---一个 (指示符[indicator]的) 正规列表[proper list].
      indicator---一个对象[object], 它是 indicator-list 中的元素[element].
      value---一个对象[object].
      tail---一个列表[list].
    
  • 描述(Description):

      get-properties 被用来同时查找多个属性列表[property list]条目中的任何一个.
    
      它在 plist 中搜索第一个指示符[indicator]和 indicator-list 中的其中一个对象相同[identical]的条目. 如果找到这样一个条目, 返回的 indicator 和 value 是这个指示符[property indicator]和它关联的属性值[property value], 而返回的 tail 是以找到的条目开始的 plist 的尾部[tail] (换句话说, 它的 car 就是那个 indicator). 如果没有找到这样的条目, 那么这个 indicator, value, 和 tail 都是 nil.
    
  • 示例(Examples):

    (setq x '()) =>  NIL
    (setq *indicator-list* '(prop1 prop2)) =>  (PROP1 PROP2)
    (getf x 'prop1) =>  NIL
    (setf (getf x 'prop1) 'val1) =>  VAL1
    (eq (getf x 'prop1) 'val1) =>  true
    (get-properties x *indicator-list*) =>  PROP1, VAL1, (PROP1 VAL1)
    x =>  (PROP1 VAL1)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      get, getf
    
  • 注意(Notes): None.

访问器 GETF

  • 语法(Syntax):

      getf plist indicator &optional default => value
    
      (setf (getf place indicator &optional default) new-value)
    
  • 参数和值(Arguments and Values):

      plist---一个属性列表[property list].
      place---一个位置[place], 它的值[value]是一个属性列表[property list].
      indicator---一个对象[object].
      default---一个对象[object]. 默认是 nil.
      value---一个对象[object].
      new-value---一个对象[object].
    
  • 描述(Description):

      getf 在 plist 上查找一个属性指示符[property indicator]等于 indicator 的属性[property], 并且返回对应的属性值[property value]. 如果这里有多个带有那个属性指示符[property indicator]的属性[property[1]], getf 使用第一个这样的属性[property]. 如果这里没有带有那个属性指示符[property indicator]的属性[property], 返回 default.
    
      getf 的 setf 可能被用于关联一个新的对象[object]到由位置 place 持有的属性列表[proper list]中的一个已存在的指示符, 或者如果不存在就创建一个新的关联. 如果这里有多个带有那个属性指示符[property indicator]的属性[property[1]], getf 的 setf 会关联这个新值 new-value 到第一个这样的属性[property]. 当一个 getf 表达式形式[form]被用作一个 setf 的 place 时, 被提供的任何 default 根据正常的从左到右的求值规则被求值, 但是它的值[value]会被忽略.
    
      getf 的 setf 允许去写入[write] place 自身的值[value], 或者修改 place 所持有的列表结构[list structure]的任意部分, car 或 cdr.
    
  • 示例(Examples):

    (setq x '()) =>  NIL
    (getf x 'prop1) =>  NIL
    (getf x 'prop1 7) =>  7
    (getf x 'prop1) =>  NIL
    (setf (getf x 'prop1) 'val1) =>  VAL1
    (eq (getf x 'prop1) 'val1) =>  true
    (getf x 'prop1) =>  VAL1
    (getf x 'prop1 7) =>  VAL1
    x =>  (PROP1 VAL1)
    
    ;; Examples of implementation variation permitted.
    (setq foo (list 'a 'b 'c 'd 'e 'f)) =>  (A B C D E F)
    (setq bar (cddr foo)) =>  (C D E F)
    (remf foo 'c) =>  true
    foo =>  (A B E F)
    bar
    =>  (C D E F)
    OR=>  (C)
    OR=>  (NIL)
    OR=>  (C NIL)
    OR=>  (C D)
    
  • 副作用(Side Effects): None.

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

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

  • 参见(See Also):

      get, get-properties, setf, 章节 5.1.2.2 (函数调用表达式形式作为位置)
    
  • 注意(Notes):

      这里无法 (使用 getf) 去区分一个省略的属性和一个值为 default 的属性; 但是见 get-properties.
    
      注意, 在一个 setf 的情况中给 getf 提供一个默认值 default 有时候不是非常有趣的, 但这仍然是很重要的, 因为某些宏, 例如 push 和 incf, 需要一个数据可以被读取[read]和写入[write]的 place 参数. 在这样一个上下文中, 如果一个 default 参数被提供用于读取[read]的情况, 对于写入[write]的情况它必须也是语法上有效的. 比如,
    
    (let ((plist '()))
      (incf (getf plist 'count 0))
      plist) =>  (COUNT 1)
    

宏 REMF

  • 语法(Syntax):

      remf place indicator => generalized-boolean
    
  • 参数和值(Arguments and Values):

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

      remf 从存储在 place 中的属性列表[property list]中移除一个属性指示符[property indicator]和 indicator 相同[identical]的属性[property[1]]. 如果这里有多个带有相同[identical]键的属性[property[1]], remf 只移除第一个这样的属性[property]. 如果没有找到这样的属性[property] remf 就返回 false, 如果找到一个属性就返回 true.
    
      这个属性指示符[property indicator]和对应属性值[property value]会通过破坏性地拼接这个属性列表以未定义的顺序被移除. remf 允许去 setf place 或者去 setf place 所持有的列表结构[list structure]的任何部分, car 或 cdr.
    
      关于 place 的子表达式形式[subform]的求值[evaluation]的信息, 见章节 5.1.1.1 (位置的子表达式形式求值).
    
  • 示例(Examples):

    (setq x (cons () ())) =>  (NIL)
    (setf (getf (car x) 'prop1) 'val1) =>  VAL1
    (remf (car x) 'prop1) =>  true
    (remf (car x) 'prop1) =>  false
    
  • 副作用(Side Effects):

      存储在 place 中的属性列表会被修改.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      remprop, getf
    
  • 注意(Notes): None.

函数 INTERSECTION, NINTERSECTION

  • 语法(Syntax):

      intersection list-1 list-2 &key key test test-not => result-list
    
      nintersection list-1 list-2 &key key test test-not => result-list
    
  • 参数和值(Arguments and Values):

      list-1---一个正规列表[proper list].
      list-2---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      result-list---一个列表[list].
    
  • 描述(Description):

      intersection 和 nintersection 返回一个包含了同时出现在 list-1 和 list-2 中的每个元素的列表[list].
    
      nintersection 是 intersection 的破坏性的版本. 它执行相同的操作, 但是可能破坏 list-1 使用它的存储格来构造这个结果. list-2 不会被破坏.
    
      这个 intersection 操作符描述如下. 对于所有可能由 list-1 的一个元素[element]和 list-2 的一个元素[element]构成的有序对, :test 或 :test-not 被用于确定它们是否满足测试条件[satisfy the test]. 给 :test 或 :test-not 函数的第一个参数是 list-1 的一个元素; 第二个参数是 list-2 的一个元素. 如果没有提供 :test 或 :test-not, 就使用 eql. 如果在同一个函数调用中提供了 :test 和 :test-not 那么就是一个错误.
    
      如果提供了 :key (并且不是 nil), 它被用于从这个列表 list 元素中提取这个要被测试的部分. 给这个 :key 函数的参数是 list-1 或 list-2 的一个元素; 这个 :key 函数通常返回提供的元素的部分. 如果没有提供 :key 或者是 nil, 那么就使用这个 list-1 和 list-2 的元素.
    
      对于满足测试条件[satisfy the test]的每一个对, 这个对的两个元素中只有一个会被放到结果中. 一个列表[list]中的元素如果和另一个列表[list]中的元素都不满足测试条件[satisfy the test]就不会出现在结果中. 如果其中一个列表[list]包含了重复的元素, 那么在结果中也可能会重复.
    
      不保证元素出现在结果中的顺序会以任何特定方式反映参数的顺序. 如果合适的话结果列表可能和 list-1 或 list-2 共享存储格, 或者和 list-1 或 list-2 是 eq 的.
    
  • 示例(Examples):

    (setq list1 (list 1 1 2 3 4 a b c "A" "B" "C" "d")
          list2 (list 1 4 5 b c d "a" "B" "c" "D")) 
      =>  (1 4 5 B C D "a" "B" "c" "D")
    (intersection list1 list2) =>  (C B 4 1 1)
    (intersection list1 list2 :test 'equal) =>  ("B" C B 4 1 1)
    (intersection list1 list2 :test #'equalp) =>  ("d" "C" "B" "A" C B 4 1 1) 
    (nintersection list1 list2) =>  (1 1 4 B C)
    list1 =>  implementation-dependent ;e.g.,  (1 1 4 B C)
    list2 =>  implementation-dependent ;e.g.,  (1 4 5 B C D "a" "B" "c" "D")
    (setq list1 (copy-list '((1 . 2) (2 . 3) (3 . 4) (4 . 5))))
    =>  ((1 . 2) (2 . 3) (3 . 4) (4 . 5)) 
    (setq list2 (copy-list '((1 . 3) (2 . 4) (3 . 6) (4 . 8))))
    =>  ((1 . 3) (2 . 4) (3 . 6) (4 . 8)) 
    (nintersection list1 list2 :key #'cdr) =>  ((2 . 3) (3 . 4)) 
    list1 =>  implementation-dependent ;e.g.,  ((1 . 2) (2 . 3) (3 . 4)) 
    list2 =>  implementation-dependent ;e.g.,  ((1 . 3) (2 . 4) (3 . 6) (4 . 8)) 
    
  • 副作用(Side Effects):

      nintersection 可以修改 list-1, 但不能修改 list-2.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list-1 和 list-2 不是正规列表[proper list], 那么应该准备发出 type-error 类型[type]的错误.
    
  • 参见(See Also):

      union, 章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数已经被废弃
    
      因为这个 nintersection 副作用不是必须的, 它不应该被用于仅为了副作用(for-effect-only)的可移植代码中.
    

函数 ADJOIN

  • 语法(Syntax):

      adjoin item list &key key test test-not => new-list
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      list---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[functioni]的标识符[designator].
      key---一个单实参[argument]函数[function]的标识符[designator], 或者 nil.
      new-list---一个列表[list].
    
  • 描述(Description):

      测试 item 是否和列表 list 中的一个已存在的元素相同. 如果 item 不是一个已存在的元素, adjoin 把它添加到 list (就好像是通过 cons 一样) 并且返回产生的列表[list]; 否则, 没有东西会被添加并且原始的列表 list 会被返回.
    
      这个 test, test-not, 和 key 影响如何决定那个 item 是否和 list 中的一个元素[element]相同. 关于详情, 见章节 17.2.1 (满足一个两个参数的测试).
    
  • 示例(Examples):

    (setq slist '()) =>  NIL 
    (adjoin 'a slist) =>  (A) 
    slist =>  NIL 
    (setq slist (adjoin '(test-item 1) slist)) =>  ((TEST-ITEM 1)) 
    (adjoin '(test-item 1) slist) =>  ((TEST-ITEM 1) (TEST-ITEM 1)) 
    (adjoin '(test-item 1) slist :test 'equal) =>  ((TEST-ITEM 1)) 
    (adjoin '(new-test-item 1) slist :key #'cadr) =>  ((TEST-ITEM 1)) 
    (adjoin '(new-test-item 1) slist) =>  ((NEW-TEST-ITEM 1) (TEST-ITEM 1)) 
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list 不是一个正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

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

      这个 :test-not 参数已经被废弃
    
      (adjoin item list :key fn)
        ==  (if (member (fn item) list :key fn) list (cons item list))
    

宏 PUSHNEW

  • 语法(Syntax):

      pushnew item place &key key test test-not
      => new-place-value
    
  • 参数和值(Arguments and Values):

      item---一个对象[object].
      place---一个位置[place], 它的值是一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      new-place-value---一个列表[list] (这个值[place]的新的值).
    
  • 描述(Description):

      pushnew 测试 item 是否和存储在位置 place 的列表[list]中的一个已存在元素相同. 如果 item 没有和任何一个元素相同的话, 它就会被追加到 list 前面, 并且这个新的列表[list]被存储到位置 place 中.
    
      pushnew 返回一个存储在 place 中的新列表[list].
    
      item 是否已经为 place 中的列表[list]的成员是通过使用 :test 或 :test-not 比较决定的. 给 :test 或 :test-not 函数的第一个参数是 item; 第二个参数是 place 中的列表[list]的元素通过 :key 函数返回的 (如果提供的话).
    
      如果提供了 :key, 它被用于提取 item 和列表[list]元素中要被测试的部分, 就像 adjoin 一样.
    
      给 :key 函数的参数是存储在 place 中的列表[list]的元素. 这个 :key 函数通常返回列表[list]的元素的一部分. 如果没有提供 :key 或者是 nil, 就是那个列表[list]元素.
    
      关于位置 place 的子表达式形式[subform]求值[evaluation]的信息, 见章节 5.1.1.1 (位置的子表达式形式求值).
    
      在这个 item 已经是 place 所持有的列表[list]的成员的情况下, pushnew 事实上是否会为它的 place 执行存储表达式形式是依赖于具体实现的[implementation-dependent].
    
  • 示例(Examples):

    (setq x '(a (b c) d)) =>  (A (B C) D)
    (pushnew 5 (cadr x)) =>  (5 B C)   
    x =>  (A (5 B C) D)
    (pushnew 'b (cadr x)) =>  (5 B C)  
    x =>  (A (5 B C) D)
    (setq lst '((1) (1 2) (1 2 3))) =>  ((1) (1 2) (1 2 3))
    (pushnew '(2) lst) =>  ((2) (1) (1 2) (1 2 3))
    (pushnew '(1) lst) =>  ((1) (2) (1) (1 2) (1 2 3))
    (pushnew '(1) lst :test 'equal) =>  ((1) (2) (1) (1 2) (1 2 3))
    (pushnew '(1) lst :key #'car) =>  ((1) (2) (1) (1 2) (1 2 3)) 
    
  • 副作用(Side Effects):

      这个 place 的内容可能被修改.
    
  • 受此影响(Affected By): None.

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

  • 参见(See Also):

      push, adjoin, 章节 5.1 (广义引用)
    
  • 注意(Notes):

      这个
    
      (pushnew item place :test p)
    
      的效果近似等价于
    
      (setf place (adjoin item place :test p))
    
      除了那个 place 的子表达式形式[subform]只会被求值一次, 并且 item 在 place 之前被求值. 
    

函数 SET-DIFFERENCE, NSET-DIFFERENCE

  • 语法(Syntax):

      set-difference list-1 list-2 &key key test test-not => result-list
    
      nset-difference list-1 list-2 &key key test test-not => result-list
    
  • 参数和值(Arguments and Values):

      list-1---一个正规列表[proper list].
      list-2---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      result-list---一个列表[list].
    
  • 描述(Description):

      set-difference 返回一个在 list-1 中的但是没有出现在 list-2 中的元素的列表[list].
    
      nset-difference 是 set-difference 的破坏性版本. 它可能破坏 list-1.
    
      对于由 list-1 的一个元素和 list-2 的一个元素组成的所有可能的有序对, 这个 :test 或 :test-not 函数被用于确定它们是否满足测试条件[satisfy the test]. 给 :test 或 :test-not 函数的第一个参数是由 :key 函数(如果提供的话)返回的 list-1 的元素的一部分; 给 :test 或 :test-not 函数的第二个参数是由 :key 函数(如果提供的话)返回的 list-2 的元素的一部分.
    
      如果提供了 :key, 它的参数就是一个 list-1 或 list-2 的元素. 这个 :key 函数通常返回提供的元素的一部分. 如果没有提供 :key, 就使用 list-1 或 list-2 的元素.
    
      当且仅当一个 list-1 中的元素不匹配任何 list-2 中的元素时它会出现在结果中.
    
      这里不保证出现在结果中的顺序会以任何特定方式反映参数的顺序. 如果合适的话, 结果列表[list]可能和 list-1 或 list-2 共享存储格, 或者和 list-1 或 list-2 是 eq 的.
    
  • 示例(Examples):

    (setq lst1 (list "A" "b" "C" "d")
          lst2 (list "a" "B" "C" "d")) =>  ("a" "B" "C" "d")
    (set-difference lst1 lst2) =>  ("d" "C" "b" "A")
    (set-difference lst1 lst2 :test 'equal) =>  ("b" "A")
    (set-difference lst1 lst2 :test #'equalp) =>  NIL 
    (nset-difference lst1 lst2 :test #'string=) =>  ("A" "b")
    (setq lst1 '(("a" . "b") ("c" . "d") ("e" . "f")))
    =>  (("a" . "b") ("c" . "d") ("e" . "f")) 
    (setq lst2 '(("c" . "a") ("e" . "b") ("d" . "a")))
    =>  (("c" . "a") ("e" . "b") ("d" . "a")) 
    (nset-difference lst1 lst2 :test #'string= :key #'cdr)
    =>  (("c" . "d") ("e" . "f")) 
    lst1 =>  (("a" . "b") ("c" . "d") ("e" . "f")) 
    lst2 =>  (("c" . "a") ("e" . "b") ("d" . "a")) 
    
    ;; Remove all flavor names that contain "c" or "w".
    (set-difference '("strawberry" "chocolate" "banana"
                      "lemon" "pistachio" "rhubarb")
              '(#\c #\w)
              :test #'(lambda (s c) (find c s)))
    =>  ("banana" "rhubarb" "lemon")    ;One possible ordering.
    
  • 副作用(Side Effects):

      nset-difference 可能破坏 list-1.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list-1 和 list-2 不是正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数已经被废弃 
    

函数 SET-EXCLUSIVE-OR, NSET-EXCLUSIVE-OR

  • 语法(Syntax):

      set-exclusive-or list-1 list-2 &key key test test-not => result-list
    
      nset-exclusive-or list-1 list-2 &key key test test-not => result-list
    
  • 参数和值(Arguments and Values):

      list-1---一个正规列表[proper list].
      list-2---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      result-list---一个列表[list].
    
  • 描述(Description):

      set-exclusive-or 返回一个只出现在 list-1 和 list-2 其中之一的元素的列表.
    
      nset-exclusive-or 是 set-exclusive-or 的破坏性版本.
    
      对于由 list-1 的一个元素和 list-2 的一个元素组成的所有可能的有序对, 这个 :test 或 :test-not 函数被用于确定它们是否满足测试条件[satisfy the test].
    
      如果提供了 :key, 它被用于从 list-1 or list-2 元素中提取要被测试的部分. 给 :test 或 :test-not 函数的第一个参数是由 :key 函数(如果提供的话)返回的 list-1 的元素的一部分; 给 :test 或 :test-not 函数的第二个参数是由 :key 函数(如果提供的话)返回的 list-2 的元素的一部分. 如果没有提供 :key 或者是 nil, 就使用 list-1 或 list-2 元素.
    
      这个结果准确包含了 list-1 和 list-2 中没有匹配对的元素.
    
      这个 set-exclusive-or 的结果列表可能和 list-1 或 list-2 其中一个共享存储.
    
  • 示例(Examples):

    (setq lst1 (list 1 "a" "b")
          lst2 (list 1 "A" "b")) =>  (1 "A" "b")
    (set-exclusive-or lst1 lst2) =>  ("b" "A" "b" "a")
    (set-exclusive-or lst1 lst2 :test #'equal) =>  ("A" "a")
    (set-exclusive-or lst1 lst2 :test 'equalp) =>  NIL 
    (nset-exclusive-or lst1 lst2) =>  ("a" "b" "A" "b") 
    (setq lst1 (list (("a" . "b") ("c" . "d") ("e" . "f"))))
    =>  (("a" . "b") ("c" . "d") ("e" . "f"))
    (setq lst2 (list (("c" . "a") ("e" . "b") ("d" . "a"))))
    =>  (("c" . "a") ("e" . "b") ("d" . "a")) 
    (nset-exclusive-or lst1 lst2 :test #'string= :key #'cdr)
    =>  (("c" . "d") ("e" . "f") ("c" . "a") ("d" . "a")) 
    lst1 =>  (("a" . "b") ("c" . "d") ("e" . "f"))
    lst2 =>  (("c" . "a") ("d" . "a")) 
    
  • 副作用(Side Effects):

      nset-exclusive-or 允许去修改 list-1 或 list-2 的列表结构[list structure]的任意部分, car 或 cdr.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list-1 和 list-2 不是正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数已经被废弃
    
      因为这个 nset-exclusive-or 副作用不是必须的, 所以它不应该被用于可移植代码中仅为了副作用(for-effect-only)的位置. 
    

函数 SUBSETP

  • 语法(Syntax):

      subsetp list-1 list-2 &key key test test-not => generalized-boolean
    
  • 参数和值(Arguments and Values):

      list-1---一个正规列表[proper list].
      list-2---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      generalized-boolean---一个广义 boolean [generalized boolean].
    
  • 描述(Description):

      如果 list-1 中的每个元素都匹配 list-2 中的某个元素 subsetp 就返回 true, 否则返回 false.
    
      一个列表元素是否和另一个列表元素相同是由关键字参数指定的函数决定的. 给 :test 或 :test-not 函数的第一个参数是由 :key 函数提取的 list-1 的元素的一部分; 给 :test 或 :test-not 函数的第二个参数是由 :key 函数提取的 list-2 的元素的一部分.
    
      给 :key 函数的参数是 list-1 或 list-2 的一个元素; 返回值是那个提供的列表元素的一部分. 如果没有提供 :key 或者是 nil, 那么这个 list-1 或 list-2 元素自身会被提供给 :test 或 :test-not 函数.
    
  • 示例(Examples):

    (setq cosmos '(1 "a" (1 2))) =>  (1 "a" (1 2))
    (subsetp '(1) cosmos) =>  true
    (subsetp '((1 2)) cosmos) =>  false
    (subsetp '((1 2)) cosmos :test 'equal) =>  true
    (subsetp '(1 "A") cosmos :test #'equalp) =>  true
    (subsetp '((1) (2)) '((1) (2))) =>  false
    (subsetp '((1) (2)) '((1) (2)) :key #'car) =>  true
    
  • 副作用(Side Effects): None.

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

  • 异常情况(Exceptional Situations):

      如果 list-1 和 list-2 不是正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

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

      这个 :test-not 参数已经被废弃 
    

函数 UNION, NUNION

  • 语法(Syntax):

      union list-1 list-2 &key key test test-not => result-list
    
      nunion list-1 list-2 &key key test test-not => result-list
    
  • 参数和值(Arguments and Values):

      list-1---一个正规列表[proper list].
      list-2---一个正规列表[proper list].
      test---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      test-not---返回一个广义 boolean [generalized boolean]的两实参[argument]函数[function]的标识符[designator].
      key---一个单参数函数[function]的标识符[designator], 或者 nil.
      result-list---一个列表[list].
    
  • 描述(Description):

      union 和 nunion 返回一个包含出现在 list-1 或 list-2 中的每一个元素的列表.
    
      对于由 list-1 的一个元素和 list-2 的一个元素组成的所有可能的有序对, 这个 :test 或 :test-not 函数被用于确定它们是否满足这个测试条件 test. 给 :test 或 :test-not 函数的第一个参数是由 :key 函数(如果提供的话)返回的 list-1 的元素的一部分; 给 :test 或 :test-not 函数的第二个参数是由 :key 函数(如果提供的话)返回的 list-2 的元素的一部分.
    
      给 :key 函数的参数是 list-1 或 list-2 的一个元素; 返回值是提供的那个元素的一部分. 如果没有提供 :key 或者是 nil, 那么这个 list-1 或 list-2 元素自身会被提供给 :test 或 :test-not 函数.
    
      对于每一个匹配序对, 这个序对的两个元素中的一个会出现在结果中. 任何来自于 list-1 或 list-2 并且在另一个中没有匹配元素的元素会出现在结果中.
    
      如果在 list-1 和 list-2 之间有重复的, 这些重复实例中只有一个会出现在结果中. 如果 list-1 或 list-2 其中有着重复的条目, 那么这些重复的条目可能会也可能不会出现在结果中.
    
      结果中元素的顺序不需要去以任何方式返回 list-1 或 list-2 中的顺序. 如果合适的话这个结果列表可能和 list-1 或 list-2 是 eq 的.
    
  • 示例(Examples):

    (union '(a b c) '(f a d))
    =>  (A B C F D)
    OR=>  (B C F A D)
    OR=>  (D F A B C)
    (union '((x 5) (y 6)) '((z 2) (x 4)) :key #'car)
    =>  ((X 5) (Y 6) (Z 2))
    OR=>  ((X 4) (Y 6) (Z 2))
    
    (setq lst1 (list 1 2 '(1 2) "a" "b")
          lst2 (list 2 3 '(2 3) "B" "C"))
    =>  (2 3 (2 3) "B" "C")
    (nunion lst1 lst2)
    =>  (1 (1 2) "a" "b" 2 3 (2 3) "B" "C") 
    OR=>  (1 2 (1 2) "a" "b" "C" "B" (2 3) 3)
    
  • 副作用(Side Effects):

      nunion 允许去修改 list-1 或 list-2 的列表结构[list]的任意部分, car 或 cdr.
    
  • 受此影响(Affected By): None.

  • 异常情况(Exceptional Situations):

      如果 list-1 和 list-2 不是正规列表[proper list]那么应该准备发出一个 type-error 类型[type]的错误.
    
  • 参见(See Also):

      intersection, 章节 3.2.1 (编译器术语), 章节 3.6 (遍历规则和副作用)
    
  • 注意(Notes):

      这个 :test-not 参数已经被废弃
    
      因为 nunion 副作用不是必须的, 所以它不应该被用于可移植代码中仅为了副作用(for-effect-only)位置. 
    
;