Clojure 的点特殊形式怪异

Clojure's dot special form weirdness

我很好奇为什么会这样(正如我在阅读 the documentation on the dot special form 后所期望的那样):

(map #(. % isInstance {}) [clojure.lang.IPersistentMap])

returns:

(true)

但这不是:

(. clojure.lang.IPersistentMap isInstance {})

我收到一个错误,"No matching method: isInstance"。形式与上面的 map 函数调用完全相同,但在 map 之外,它不起作用。为什么?

.(点)系列形式是其中最奇怪的。不确定我是否会解释得很好,但让我们试试吧。

根据 docs:

If the first operand is a symbol that resolves to a class name, the access is considered to be to a static member of the named class. Note that nested classes are named EnclosingClass$NestedClass, per the JVM spec. Otherwise it is presumed to be an instance member and the first argument is evaluated to produce the target object.

强调我的。

所以您遇到了第一种情况 (. clojure.lang.IPersistentMap isInstance {}) - clojure.lang.IPersistentMap 解析为 class 名称,整个表达式被假定为静态方法调用。

map 的情况下,符号被求值(强调的部分),在传递给匿名函数之前求值为 clojure.lang.IPersistentMap class 对象,整个表达式是一个class.

上的实例方法调用

所以它归结为这样一个事实,即在一个地方 clojure.lang.IPersistentMap 被用作引用 class 名称的符号,而在另一个地方用作计算结果为 [= 的东西34=] 对象。

另见here:

Note that class names normally denote class objects, but are treated specially in certain special forms, e.g. '.' and new.