Группировать несколькими клавишами в Clojure

Как сгруппировать коллекцию карт несколькими ключами?

Например:

(def m1 [{:a 1 :b 2 :c 3}
         {:a 1 :b 2 :c 4}
         {:a 1 :b 4 :c 3}
         {:a 1 :b 4 :c 3}])

(group-by-x [:a :b] m1)

Я хотел бы вернуть это:

[{:a 1 :b 2} [{:a 1 :b 2 :c 3}{:a 1 :b 2 :c 4}],
 {:a 1 :b 4} [{:a 1 :b 4 :c 3}{:a 1 :b 4 :c 3}]]

Ответы

Ответ 1

(group-by #(select-keys % [:a :b]) m1)

Это возвращает карту:

{{:b 2, :a 1} [{:a 1, :c 3, :b 2} {:a 1, :c 4, :b 2}],
 {:b 4, :a 1} [{:a 1, :c 3, :b 4} {:a 1, :c 3, :b 4}]}

Чтобы получить именно указанное вами возвращаемое значение, оберните его в (vec (apply concat ...)):

(vec (apply concat (group-by #(select-keys % [:a :b]) m1)))
; => as in the question text

Это эквивалентно, но, возможно, красивее:

(->> (group-by #(select-keys % [:a :b]) m1)
     (apply concat)
     vec)