1

我是一个 Common Lisp 初学者,遇到了这段代码:

(let ((foo (list 42)))
  (setf (rest foo) foo))

REPL 在尝试执行时似乎只是永远循环。

4

1 回答 1

8

是什么FOO

FOO最初是一个新列表,(42). 在 Lisp 中,列表由cons cellsCAR表示,即包含每个 a和一个CDRslot的可变内存块。另一种打印方式是(42 . NIL)CARCDR位于点的每一侧。这也可以如下图所示:

  car  cdr
------------
| 42 | NIL |
------------
     ^
     |
    FOO

当您SETF使用(rest foo) 位置foo值调用时,您是说您希望cdr单元格FOO保存该值FOO。事实上,这个 的出现SETF很可能会宏扩展为对 的调用RPLACD

------------
| 42 | FOO |
------------
     ^
     |
    FOO

为什么 REPL 永远循环?

“REPL”(打印)的“P”部分尝试打印您的圆形结构。这是因为SETF' 的值是从被评估的表单返回的值,而返回SETF的值是其第二个参数的值,即FOO。想象一下,你想用一个简单的算法编写一个 cons cell X:

1. PRINT "("
2. PRINT the CAR of X
3. PRINT " . "
4. PRINT the CDR of X
5. PRINT ")"

但是,对于foo,第 4 步将递归打印相同的结构,并且永远不会终止。

你能做什么?

首先尝试设置*PRINT-CIRCLE*为 T:

(setf *print-circle* t)

现在,您的 REPL 应该很高兴:

CL-USER> (let ((foo (list 42)))
           (setf (rest foo) foo))
#1=(42 . #1#)

Sharpsign 等号”表示法允许读者将表单的一部分影响到(读者)变量,例如#1=...,并在之后重用它,例如#1#。这使得在读取或打印期间表示数据之间的循环交叉引用成为可能。在这里,我们可以看到变量#1#表示一个 cons-cell,其中CAR是 42 而CDR#1#本身。

于 2016-12-02T00:16:53.983 回答