Clojure 中的 :foo、::foo、::bar/foo 和 :bar/foo 之间有什么区别?

What is the difference between :foo, ::foo, ::bar/foo, and :bar/foo in Clojure?

我遇到了这些关键字::foo, ::foo, ::bar/foo, and :bar/foo

能否举例说明差异?

This article 解释得很好。

TL;DR:

  • :foo是没有命名空间的关键字,即非限定关键字。
  • :foo/bar 是具有显式命名空间 foo.
  • 的限定关键字
  • ::foo 是自动限定关键字。如果此关键字出现在命名空间 bar 中,则其计算结果为 :bar/foo.
  • ::bar/foo的命名空间与:baz/foo相同,如果你需要bar :as baz的命名空间。

在代码中:

boot.user=> (ns foo)
nil
foo=> (ns bar (:require [foo :as f]))
nil
bar=> (def examples [:foo ::foo ::f/foo :bar/foo])
#'bar/examples
bar=> examples
[:foo :bar/foo :foo/foo :bar/foo]
bar=> (use 'clojure.pprint)
nil
bar=> (print-table (map (fn [example] 
                          {:key example :namespace (namespace example)
                           :name (name example)}) 
                     examples))

|     :key | :namespace | :name |
|----------+------------+-------|
|     :foo |            |   foo |
| :bar/foo |        bar |   foo |
| :foo/foo |        foo |   foo |
| :bar/foo |        bar |   foo |

:foo 是一个 non-fully-qualified 关键字。它没有关联的命名空间。

(name :foo)            ; => "foo"
(namespace :foo)       ; => nil

:bar/foo 是一个 namespace-qualified 关键字。命名空间为bar,关键字名称为foo

(name :bar/foo)        ; => "foo"
(namespace :bar/foo)   ; => "bar"

::foo 会将当前命名空间与关键字相关联,使其完全限定。假设当前命名空间是 "user":

*ns*                   ; => #namespace[user]
(name ::foo)           ; => "foo"
(namespace ::foo)      ; => "user"
::foo                  ; => :user/foo

::bar/foo 将尝试扩展任何别名以使其完全合格:

(create-ns 'my.namespace)       ; => #namespace[my.namespace]
(alias 'bar 'my.namespace)      ; => nil
(name ::bar/foo)                ; => "foo"
(namespace ::bar/foo)           ; => "my.namespace"
::bar/foo                       ; => :my.namespace/foo