В чем разница между DoSq и for в Clojure? Каковы примеры того, когда вы бы предпочли использовать одно вместо другого?
В чем разница между DoSq и for в Clojure? Каковы примеры того, когда вы бы предпочли использовать одно вместо другого?
Ответы:
Разница в том, что for
строит ленивую последовательность и возвращает ее, а doseq
предназначена для выполнения побочных эффектов и возвращает nil.
user=> (for [x [1 2 3]] (+ x 5))
(6 7 8)
user=> (doseq [x [1 2 3]] (+ x 5))
nil
user=> (doseq [x [1 2 3]] (println x))
1
2
3
nil
Если вы хотите создать новую последовательность на основе других последовательностей, используйте for. Если вы хотите создать побочные эффекты (печать, запись в базу данных, запуск ядерной боеголовки и т. Д.) На основе элементов из некоторых последовательностей, используйте дозу.
Также обратите внимание, что doseq
любит, когда for
ленится. Пример, отсутствующий в ответе Рейна:
(for [x [1 2 3]] (println x))
В REPL это обычно будет делать то, что вы хотите, но это в основном совпадение: REPL заставляет ленивую последовательность for
, созданную с помощью , вызывая выполнение printlns. В неинтерактивной среде ничего не будет напечатано. Вы можете увидеть это в действии, сравнив результаты
user> (def lazy (for [x [1 2 3]] (println 'lazy x)))
#'user/lazy
user> (def eager (doseq [x [1 2 3]] (println 'eager x)))
eager 1
eager 2
eager 3
#'user/eager
Поскольку def
форма возвращает созданную новую переменную, а не значение, которое к ней привязано, REPL нечего печатать и lazy
будет ссылаться на нереализованный lazy-seq: ни один из его элементов не был вычислен вообще. eager
будет ссылаться на nil
, и вся его печать будет выполнена.