Как я могу обновить векторный объект в Clojure?

Данный:

(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 за исключением заменяющих значений.

12
задан nha 16 August 2015 в 16:27
поделиться

3 ответа

сопоставляет функцию с вектором карт, которая либо создает измененную карту, если ключ совпадает, либо использует исходную, если ключи не совпадают, затем преобразует результат обратно в вектор

(vec (map #(if (= (:id %) 1) 
             (assoc % :a "baz2" :b "spam2")
             %)))

Это можно сделать еще лаконично, хотя это действительно показывает, где происходит структурное разделение.

8
ответ дан 2 December 2019 в 07:21
поделиться

Знаете ли вы заранее, что карта с 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"}}
10
ответ дан 2 December 2019 в 07:21
поделиться

Возможно, вы захотите взглянуть на array-map , который создает карту, поддерживаемую массивом и снабженную индексом, вместо использования :я бы?

3
ответ дан 2 December 2019 в 07:21
поделиться
Другие вопросы по тегам:

Похожие вопросы: