2024年Clojure状态调查!中分享您的想法。

欢迎!请查看关于页面以获取更多关于这个功能的信息。

+1 投票
Clojure

以下表达式打印{{1234}}并返回{{1}}

(first (mapcat #(do (print %) [%]) '(1 2 3 4 5 6 7)))

原因是{{(apply concat args)}}在其参数中并不是最懒惰的,并且实际上会在返回第一个项目之前实现前四个。这对于变长{{concat}}来说是不可避免的。

这可以通过仅修改{{mapcat}}来修复,或者通过添加一个新的函数(到{{clojure.core}}?)来修复,该函数是{{concat}}的非变长等效函数,并使用它重新实现{{mapcat}}。

(defn join "懒惰地将序列-of-序列连接成一个扁平序列。" [s] (lazy-seq (when-let [[x & xs] (seq s)] (concat x (join xs)))))

6 个答案

0 投票
_由:gfredericks_发表的评论

我意识到,如果{{concat}}有一个单条的{{[& args]}},则可以不改变其语义,实际上可以使其更懒惰。
0 投票

由:eigenhombre发表的评论

我花了几个小时在上个月理解这个问题(链接:1,2),然后今天在Jira中看到这个工单... +1。

(link: 1) http://eigenhombre.com/2013/07/13/updating-the-genome-decoder-resulting-consequences/

(link: 2) http://clojurian.blogspot.com/2012/11/beware-of-mapcat.html

0 投票

评论由: gfredericks

将join代码更新为实际有效。

0 投票

评论由: gshayban

描述中使用的join版本也不具备最大程度的惰性,并且将实现底层集合的两种。原因:将seq拆解会导致对'x'的'nth'调用和对'xs'的'nthnext'调用。nthnext不是最大程度的惰性。

`
(defn join
"惰性连接序列-of-序列到一个平坦序列。"
[s]
(lazy-seq
(when-let [s (seq s)]

 (concat (first s) (join (rest s))))))

`

0 投票

评论由: gshayban

尽管帮助说明并没有做出关于惰性的承诺(除了声明其特定的实现),这似乎是一个缺陷而非增强。mapcat最少实现4个底层集合

`
boot.user=> (defn notifying-seq [cb!] (lazy-seq (cb!) (cons :ignored (notifying-seq cb!))))

'boot.user/notifying-seq

boot.user=> (let [a (atom 0)]
(seq (mapcat (constantly [:ignored :ignored])

           (notifying-seq #(swap! a inc))))

@a)
4
`

0 投票
参考: https://clojure.atlassian.net/browse/CLJ-1218 (由gfredericks报告)
...