首页 > 解决方案 > Common Lisp:为什么 delete-if if 与 remove-if 结合 setf 有很大不同?

问题描述

我试图对集合执行破坏性操作,并注意到 delete-if 在 SBCL 和 GNU CLISP 中给了我令人惊讶的结果。

将 setf 与 remove-if 结合使用可以按预期工作,但我不明白为什么 delete-if 不修改列表。

这是我的代码

(format t "trying weird delete-if problem~%")

(defun data ()
  (loop for x from 1 to 9 by 2 collect (cons x (1+ x))))

(defparameter *c1* (data))
(defparameter *c2* (data))

(format t "*c1* is now ~A~%" *c1*)
(format t "*c2* is now ~A~%" *c2*)

(delete-if
 (lambda (x)
   (progn
     (format t "trying to compare ~A ~A~%" x (equal x (cons 1 2)))
     (equal x (cons 1 2))))
 *c1*)

(format t "now trying remove-if~%")

(setf *c2* (remove-if
            (lambda (x)
              (progn
                (format t "trying to compare ~A ~A~%" x (equal x (cons 1 2)))
                (equal x (cons 1 2))))
            *c2*))

(format t "*c1* is now ~A~%" *c1*)
(format t "*c2* is now ~A~%" *c2*)

标签: common-lisp

解决方案


使用时delete-if,您仍然必须将返回值分配回保存原始列表的变量。这是因为破坏性删除可以改变列表的身份。

如果列表非空,则列表的标识是它的第一个 cons 单元格;否则它的身份是符号nil,如果它是空的。如果破坏性删除操作生成的列表是nil(当前一个列表不是时nil)或生成的列表在前面具有不同的 cons 单元格,则标识已更改。在这种情况下,必须将新身份分配回保存列表的位置。

列表的破坏性操作并不意味着我们不必为变量分配任何东西,因为列表具有引用语义。因为 Lisp 列表是对单元格的未封装引用,否则nil当我们破坏性地使用列表时,我们必须将存储列表的位置视为列表工作定义的一部分。破坏性列表不仅仅是对列表或 else 的第一个单元格的引用nil,而且破坏性列表是保存此类引用或 else 的地方nil

当破坏性地处理列表时,您几乎总是希望只有一个位置。它是列表的一部分;破坏性列表通常不应存在于两个地方。

我们可以为自己创建一个宏,用于从包含序列的地方删除,delete-if-place类似于如何处理一个地方。pushpop

最简单的方法是使用define-modify-macro; 不幸的是,该宏希望使用将位置的值作为最左侧参数的过滤函数,而delete-if将其作为第二个参数。我们可以编写一个包装函数alt-delete-if来反转两个参数:

(defun alt-delete-if (pred seq &rest args)
  (apply #'delete-if seq pred args))

alt-delete-if我们说(alt-delete-if list predicate ...)而不是(delete-if predicate list ...)

然后我们可以:

(define-modify-macro delete-if-place (&rest args) alt-delete-if)

现在我们可以这样做:

[1]> (defvar l (list 1 2 3 4 5 6))
L
[2]> (delete-if-place l #'oddp)
(2 4 6)

该地点已更新为新列表:

[3]> l
(2 4 6)

(delete-if-place place pred)执行(setf place (alt-delete-if pred place)),除了place只评估一次。


推荐阅读