我有一个嵌套的hashmap,其结构类似于以下内容:
(def pathmap {:order {:genus {:species {:cat2 "data", :cat1 "data"}}}})
Run Code Online (Sandbox Code Playgroud)
我需要的是像assoc-in这样的函数,它允许我将新的键值对添加到最里面的嵌套映射,而不是简单地替换已存在的那些.例如:
(my-assoc-merge pathmap [:order :genus :species] {:cat3 "data"})
;=> {:order {:genus {:species {:cat3 "data" :cat2 "data", :cat1 "data"}}}}
Run Code Online (Sandbox Code Playgroud)
我认为通过稍微改变assoc-in函数以使用merge-with和union,我可以很容易地做到这一点:
(defn my-assoc-merge
[m [k & ks] v]
(if ks
(assoc m k (my-assoc-merge (get m k) ks v))
(assoc m k (merge-with clojure.set/union (get m k '{}) v))))
Run Code Online (Sandbox Code Playgroud)
不幸的是,这给了我错误"CompilerException java.lang.UnsupportedOperationException:此类型不支持count:关键字,编译:(core.clj:34:12)".我哪里错了?
实际上,assoc-in如果它不存在,已经创建了一个新密钥.用法如下:
(def pathmap {:order {:genus {:species {:cat2 "data", :cat1 "data"}}}})
(assoc-in pathmap [:order :genus :species :cat3] "data")
;=> {:order {:genus {:species {:cat2 "data", :cat3 "data", :cat1 "data"}}}}
Run Code Online (Sandbox Code Playgroud)
如果您希望合并一个新的地图,使用update-in与merge
(update-in pathmap [:order :genus :species] merge {:cat3 "data"})
;=> {:order {:genus {:species {:cat2 "data", :cat3 "data", :cat1 "data"}}}}
Run Code Online (Sandbox Code Playgroud)
您的问题my-assoc-merge是union哪些适用于集合.更改merge-with到merge并删除union.