为什么这个简单的LISP函数会引发错误?

时间:2019-12-07 06:32:07

标签: syntax-error lisp common-lisp clisp

我将此函数与较大的脚本隔离开了,并通过https://www.jdoodle.com/execute-clisp-online/运行了该函数。即使抛出错误,除非我明显遗漏了一些东西,否则它似乎仍然遵循LISP的规则。

(defun cannibals-can-eat (state start-state)
    (let ((left-bank-missionaries 2)
         (left-bank-cannibals 5)
         (right-bank-missionaries (- 3 left-bank-missionaries))
         (right-bank-cannibals (- 2 left-bank-cannibals)))

         (if (or (> left-bank-cannibals left-bank-missionaries)
                 (> right-bank-cannibals right-bank-missionaries))
             t
             nil)))

错误有时是The variable LEFT-BANK-MISSIONARIES is unbound.unmatched close parenthesissyntax error near unexpected token('`。在此版本的函数中,错误是后者。

1 个答案:

答案 0 :(得分:4)

在Common Lisp中,有两种形式的本地声明(let):

(let ((var1 exp1)
      (var2 exp2)
      ...
      (varn expn))
  exp)

(let* ((var1 exp1)
       (var2 exp2)
       ...
       (varn expn))
  exp)

首先,每个表达式expilet之前的环境中求值。在第二个表达式中,每个表达式expi在包含所有先前声明var1 ... var(i-1)的环境中求值。

因此,在您的示例中,right-bank-missionaries的声明使用的left-bank-missionaries是未定义的,因为它是在同一let中声明的。

只需使用let*即可在声明每个变量后立即使用它们:

(defun cannibals-can-eat (state start-state)
    (let* ((left-bank-missionaries 2)
           (left-bank-cannibals 5)
           (right-bank-missionaries (- 3 left-bank-missionaries))
           (right-bank-cannibals (- 2 left-bank-cannibals)))

      (or (> left-bank-cannibals left-bank-missionaries)
          (> right-bank-cannibals right-bank-missionaries))))

请注意,如果您想返回generalized boolean,则最后一个if是没有用的。