8

非常简单+愚蠢的问题:

clojure 是否提供多地图?我目前有这样的事情:

(defn wrap [func]
  (fn [mp x]
    (let [k (func x)]
      (assoc mp k
             (match (get mp k)
               nil [x]
               v (cons v x))))))

(defn create-mm [func lst]
  (reduce (wrap func) {} lst))

最终创建了一个映射,对于每个键,我们都有一个包含该键的所有元素的向量。但是,看起来多图是一个非常基本的数据结构,我想知道clojure是否内置了它。

谢谢

4

1 回答 1

11

我不认为这作为一种独特的类型真的很有必要,因为 Clojure 的灵活性允许您通过使用映射和集合来快速制作自己的类型。看这里:

http://paste.lisp.org/display/89840

编辑(我应该把它粘贴进去,因为它太小了)

示例代码(由 Stuart Sierra 提供)

(ns #^{:doc "A multimap is a map that permits multiple values for each
  key.  In Clojure we can represent a multimap as a map with sets as
  values."}
  multimap
  (:use [clojure.set :only (union)]))

(defn add
  "Adds key-value pairs the multimap."
  ([mm k v]
     (assoc mm k (conj (get mm k #{}) v)))
  ([mm k v & kvs]
     (apply add (add mm k v) kvs)))

(defn del
  "Removes key-value pairs from the multimap."
  ([mm k v]
     (let [mmv (disj (get mm k) v)]
       (if (seq mmv)
         (assoc mm k mmv)
         (dissoc mm k))))
  ([mm k v & kvs]
     (apply del (del mm k v) kvs)))

(defn mm-merge
  "Merges the multimaps, taking the union of values."
  [& mms]
  (apply (partial merge-with union) mms))

(comment
  (def mm (add {} :foo 1 :foo 2 :foo 3))
  ;; mm == {:foo #{1 2 3}}

  (mm-merge mm (add {} :foo 4 :bar 2))
  ;;=> {:bar #{2}, :foo #{1 2 3 4}}

  (del mm :foo 2)
  ;;=> {:foo #{1 3}}
)

评论中指出的案例的额外测试:

(comment
  (-> {} (add :a 1) (del :a 1) (contains? :a))
  ;;=> false
)
于 2012-06-08T04:16:38.377 回答