首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Clojure中的随机树遍历

Clojure中的随机树遍历
EN

Stack Overflow用户
提问于 2016-09-28 05:42:35
回答 2查看 105关注 0票数 1

如果一棵树编码为一组嵌套列表(例如,(+ 2 (+ 2 1) (+ 1 3 2))),那么是否有一种已知的算法可以随机遍历树,在单个节点上应用一个参数提供的函数,并且在任何节点上“着陆”的概率相等?注意:在转换单个节点之后,步行就结束了。

我希望该算法的表现如下:

代码语言:javascript
复制
(def tree '(1 (1 (1 1 1) 1) 1))
(stochastic-tree-f-app inc tree) => (1 (1 (1 2 1) 1) 1)
(stochastic-tree-f-app inc tree) => (1 (1 (1 1 2) 1) 1)
(stochastic-tree-f-app inc tree) => (2 (1 (1 1 1) 1) 1)
(stochastic-tree-f-app inc tree) => (1 (1 (1 1 1) 1) 2)
(stochastic-tree-f-app dec tree) => (1 (1 (1 1 1) 0) 1)
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2016-09-28 17:07:04

使用clojure.zip

代码语言:javascript
复制
(require '[clojure.zip :as z])

(defn stochastic-tree-f-app [f tree]
  (let [zp    (z/zipper list? seq (fn [_ c] c) tree)
        nodes (->> (iterate z/next zp)
                   (take-while (complement z/end?))
                   (filter (comp integer? z/node))
                   (into []))]
    (-> (rand-nth nodes)
        (z/edit f)
        z/root)))
票数 4
EN

Stack Overflow用户

发布于 2016-09-28 14:10:23

如果可以取消最后一个需求,您可以简单地使用clojure.walk (即,...walk在转换单个节点后终止)。或者用拉链从节点上走下来,然后用编辑结束。使用clojure.walk:

代码语言:javascript
复制
(use 'clojure.walk)

(def tree '(1 (1 (1 1 1) 1) 1))

(defn stochastic-tree-f-app [f tree]
  (let [cnt (atom 0)
        _   (postwalk #(if (integer? %) (swap! cnt inc)) tree)
        idx (rand-int @cnt)]
    (reset! cnt 0)
    (postwalk #(if (and (integer? %) (= idx (swap! cnt inc)))
                (f %)
                %)
              tree)))

user> (stochastic-tree-f-app inc tree)
(2 (1 (1 1 1) 1) 1)
user> (stochastic-tree-f-app inc tree)
(1 (1 (1 1 1) 2) 1)
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/39738874

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档