Ссылка на карту против карты против рефсов против нескольких реферов - PullRequest
4 голосов
/ 17 июня 2010

Я работаю над приложением с графическим интерфейсом в Swing + Clojure, которому требуются различные изменяемые фрагменты данных (например, позиция прокрутки, пользовательские данные, имя файла, выбранные параметры инструмента и т. Д.).

Я вижу как минимум три различных способа обработки этого набора данных:

Создайте ссылку на карту всех данных:

(def data (ref {
  :filename    "filename.xml" 
  :scroll      [0 0] }))

Создайте картуиз ссылок на отдельные элементы данных:

(def datamap {
  :filename    (ref "filename.xml") 
  :scroll      (ref [0 0]) }))

Создайте отдельную ссылку для каждого в пространстве имен:

(def scroll (ref [0 0]))    

(def filename (ref "filename.xml"))

Примечание: эти данные будут доступны одновременно, например, потоками фоновой обработкиили поток обработки событий Swing.Однако, вероятно, нет необходимости в последовательных транзакционных обновлениях нескольких элементов.

Какой ваш рекомендуемый подход и почему?

Ответы [ 2 ]

6 голосов
/ 17 июня 2010

Во-первых, если вы в конечном итоге используете либо один фрагмент изменяемого состояния, либо несколько независимых фрагментов (независимо от того, что «нет смысла спрашивать, совместимо ли одно с другим»), почему бы не использовать Atoms, а не Refs?Это довольно значительное сокращение накладных расходов, которое может иметь значение, если вы делаете много обновлений.

Во-вторых, сохранение имени файла, позиции прокрутки и т. Д. В отдельных ссылках (но не в атомах) - это хорошо, пока вы разрабатываетеваши транзакции тщательно (то есть все соответствующие ссылки должны быть упомянуты, некоторые, возможно, должны быть ensure d и т. д. - в основном транзакция должна обязательно провалиться, если возникнет несогласованное состояние).Тщательный дизайн таких транзакций может быть пустой тратой усилий, если вы управляете состоянием графического интерфейса, большинство из которых почти никогда не изменяются (кроме позиции прокрутки и содержимого буфера, что на самом деле может часто меняться? - я имею в виду, что это стоит серьезно рассмотреть,В качестве ответа следует определить окончательный дизайн).Существует множество сценариев, в которых наличие нескольких объектов ссылочного типа предпочтительнее, чем одного, я просто не уверен, что одним из них является управление базовым состоянием графического интерфейса.: -)

Обратите внимание, что выполнение обновлений для вложенных структур, содержащихся в объектах ссылочного типа, очень чисто в Clojure, например (с использованием атома):

;; assuming that state-o-matic is an Atom which holds a map,
;; which holds the key of :foo, with which a vector is associated,
;; whose first item is another vector, whose first item is a map
;; which holds the key of :bar associated to a number,
;; the following increments said number
(swap! state-o-matic update-in [:foo 0 1 :bar] inc)

См. Также get-in & assoc-in.

1 голос
/ 17 июня 2010

Если нет параллелизма - тогда нет никакой разницы, вы знаете.

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

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...