让vs def in clojure

Jas*_*ker 36 lisp function clojure let

我想Scanner在clojure程序中创建一个Java 类的本地实例.为什么这不起作用:

; gives me:  count not supported on this type: Symbol 
(let s (new Scanner "a b c"))
Run Code Online (Sandbox Code Playgroud)

但它会让我创建一个像这样的全局实例:

(def s (new Scanner "a b c"))
Run Code Online (Sandbox Code Playgroud)

我的印象是唯一的区别是范围,但显然不是.let和之间有什么区别def

Ray*_*yne 53

问题是你的使用let是错误的.

让我们这样工作:

(let [identifier (expr)])
Run Code Online (Sandbox Code Playgroud)

所以你的例子应该是这样的:

(let [s (Scanner. "a b c")]
  (exprs))
Run Code Online (Sandbox Code Playgroud)

你只能在let(开启和关闭的parens)范围内使用let制作的词法绑定.让我们创建一组词法绑定.我使用def进行全局绑定,并允许在let的范围内绑定我想要的东西,因为它可以保持干净.他们都有自己的用途.

注意:( Class.)与(new Class)相同,它只是语法糖.


Sva*_*nte 33

LET不是"在当前范围内进行词法绑定",而是"使用以下绑定创建一个新的词法范围".

(let [s (foo whatever)]
  ;; s is bound here
  )
;; but not here
(def s (foo whatever))
;; s is bound here


ayr*_*ieu 12

正确的语法:

(let [s (Scanner. "a b c")] ...)
Run Code Online (Sandbox Code Playgroud)


Mar*_*rko 12

简化:def用于全局常量,let用于局部变量.

  • 不,这是一个过于简单化,这正是导致原始提问者的困惑.LET使用词法绑定创建_new_block_,而DEF仅创建新的"全局"绑定. (3认同)

Mar*_*usQ 5

即使含义相关,它们的语法也不同。

let 接受一个绑定列表(名称值对),后跟要在这些绑定的上下文中计算的表达式。

def 只接受一个绑定,而不是一个列表,并将其添加到全局上下文中。