clojure以编程方式命名空间映射键

nha*_*nha 4 namespaces clojure clojurescript

我最近namespaced maps在clojure 中学到了什么.非常方便,我想知道以编程方式命名地图的惯用方法是什么?还有其他我不知道的语法吗?

  ;; works fine
  (def m #:prefix{:a 1 :b 2 :c 3})
  (:prefix/a m) ;; 1

  ;; how to programmatically prefix the map?
  (def m {:a 1 :b 2 :c 3})
  (prn #:prefix(:foo m))        ;; java.lang.RuntimeException: Unmatched delimiter: )
Run Code Online (Sandbox Code Playgroud)

Jos*_*osh 7

这个功能可以做你想要的:

(defn map->nsmap
  [m n]
  (reduce-kv (fn [acc k v]
               (let [new-kw (if (and (keyword? k)
                                     (not (qualified-keyword? k)))
                              (keyword (str n) (name k))
                              k) ]
                 (assoc acc new-kw v)))
             {} m))
Run Code Online (Sandbox Code Playgroud)

你可以给它一个实际的命名空间对象:

(map->nsmap {:a 1 :b 2} *ns*)
=> #:user{:a 1, :b 2}

(map->nsmap {:a 1 :b 2} (create-ns 'my.new.ns))
=> #:my.new.ns{:a 1, :b 2}
Run Code Online (Sandbox Code Playgroud)

或者给它命名空间名称的字符串:

(map->nsmap {:a 1 :b 2} "namespaces.are.great")
=> #:namespaces.are.great{:a 1, :b 2}
Run Code Online (Sandbox Code Playgroud)

它只会改变非限定关键字的键,这与#:宏的行为相匹配:

(map->nsmap {:a 1, :foo/b 2, "dontalterme" 3, 4 42} "new-ns")
=> {:new-ns/a 1, :foo/b 2, "dontalterme" 3, 4 42}
Run Code Online (Sandbox Code Playgroud)