Данный:
(def my-vec [{:id 0 :a "foo" :b "bar"} {:id 1 :a "baz" :b "spam"}
{:id 2 :a "qux" :b "fred"}])
Как я могу идиоматически обновить * объект в мой-vec с :id=1
иметь значения :a="baz2"
и :b="spam2"
?
*: Я распознаю, что на самом деле не обновил бы мой-vec, но действительно возвратил бы новый вектор, который идентичен мой-vec за исключением заменяющих значений.
сопоставляет функцию с вектором карт, которая либо создает измененную карту, если ключ совпадает, либо использует исходную, если ключи не совпадают, затем преобразует результат обратно в вектор
(vec (map #(if (= (:id %) 1)
(assoc % :a "baz2" :b "spam2")
%)))
Это можно сделать еще лаконично, хотя это действительно показывает, где происходит структурное разделение.
Знаете ли вы заранее, что карта с id == 1 является второй картой в вашем векторе? Если да:
user> (-> my-vec
(assoc-in [1 :a] "baz2")
(assoc-in [1 :b] "spam2"))
[{:id 0, :a "foo", :b "bar"} {:id 1, :a "baz2", :b "spam2"} {:id 2, :a "qux", :b "fred"}]
Если вам нужно часто обращаться к данным по id, то другая идея - заменить вектор хэш-карт на хэш-карту хэш-карт с ключом :id
. Тогда вы сможете более легко assoc-in
независимо от порядка вещей.
user> (def new-my-vec (zipmap (map :id my-vec) my-vec))
#'user/new-my-vec
user> new-my-vec
{2 {:id 2, :a "qux", :b "fred"}, 1 {:id 1, :a "baz", :b "spam"}, 0 {:id 0, :a "foo", :b "bar"}}
user> (-> new-my-vec
(assoc-in [1 :a] "baz2")
(assoc-in [1 :b] "spam2"))
{2 {:id 2, :a "qux", :b "fred"}, 1 {:id 1, :a "baz2", :b "spam2"}, 0 {:id 0, :a "foo", :b "bar"}}
Возможно, вы захотите взглянуть на array-map , который создает карту, поддерживаемую массивом и снабженную индексом, вместо использования :я бы?