LISP:为什么我不能在空列表上使用 cons?

标签 lisp common-lisp

我正在尝试通过在 LISP 中使用以下代码添加到一个空列表来构造一个唯一的元素列表:

    ;;;MEMSET - return T if an atom is a top-level member of a set, else NIL
    ;;;This is needed for the makeset function
    (DEFUN MEMSET (ATM L)
        (COND ( ( NULL L) NIL )
              ( (EQL ATM(CAR L)) T )
              ( T    (MEMSET ATM (CDR L)) )
        )
    )
    (DEFUN MAKESET(SET1)
        (DO ((UNIQ ()))
            ( (NULL SET1) UNIQ) 
            (COND ( (NOT (MEMSET (CAR SET1) UNIQ))
                      (CONS (CAR SET1) UNIQ)
                  )
            )
        (SETF SET1 (CDR SET1))
        ) 
    )

当我调用 (makeset '(a b b a c d b a)) 时,这个特定代码导致 NIL - 它应该导致 (a b c d),不考虑顺序) - 但在我看来它应该从 SET1 添加一个原子还没有在每次迭代中出现在 UNIQ 中。您不能添加到在 do 循环中声明的空列表中,还是我遇到了其他问题?顺便说一下,我正在使用 clisp。

最佳答案

<强>1。使用正确的格式

询问有关 Common-Lisp 的问题时,请使用正确的格式。 例如,请参阅谷歌搜索返回的前三页 "common lisp formatting conventions" .

以下是应用于您的函数的常规范式示例:

;;; Memset - return T if an atom is a top-level member of a set, else NIL
;;; This is needed for the makeset function

(defun memset (atm l)
  (cond ((null l) nil)
        ((eql atm (car l)) t)
        (t (memset atm (cdr l)))))

(defun makeset (set1)
  (do ((uniq ()))
      ((null set1) uniq)
    (cond ((not (memset (car set1) uniq))
           (push (car set1) uniq)))
    (setf set1 (cdr set1))))

<强>2。尽可能使用原始函数

你的两个函数在 Common-Lisp 中都是原始的。

remove-duplicates返回没有重复元素的列表,member检查一个元素是否属于一个集合。

<强>3。你的函数中的错误

如果您仍然想使用您的函数,而不是 remove-duplicates,这就是其中的问题。

如果你想修改一个列表,你应该使用一个修改某些东西的函数。 cons 函数构建一个新的对,但不修改任何内容。因此,在您的表单 (cons (car set1) uniq) 中,您向 uniq 添加了一个新元素,因为您获得了一个新列表 (car set1) 作为第一个元素,unique 的元素作为列表的其余部分,但是这个新列表会立即被丢弃,因为它没有分配给任何东西。

您可以使用宏 setf 来更改它。以这种方式将新值分配给局部变量 uniq:

(setf uniq (cons (car set1) uniq))

或者您可以通过使用宏 push 来编写等效的形式:

(push (car set1) uniq)

最后,请注意 makeset 函数中的 cond 可以替换为更简洁的 when,从而产生此函数:

(defun makeset (set1)
  (do ((uniq nil))
      ((null set1) uniq)
    (when (not (memset (car set1) uniq))
      (push (car set1) uniq))
    (setf set1 (cdr set1))))

关于LISP:为什么我不能在空列表上使用 cons?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35977484/

相关文章:

lisp - 为什么CLOS slot可以解绑?

functional-programming - 具有运行时访问 AST 的编程语言/平台

loops - 在 Common Lisp 循环中定义局部变量

clojure - 在 clojure 中用宏捕获变量

optimization - 如何说服 Lisp SBCL 进行内联 fixnum 运算?

function - 将函数作为参数传递 - lambda 表达式错误?

lisp - 在 Lisp 中一次定义 n 个函数

heap-memory - 使用 SBCL 编译 lapack 系统时堆耗尽

types - Common Lisp中的类型丰富的编程?

lisp - Vlisp autocad : condition (> a b) is true even when a is equal b. 为什么?