我正在尝试将http://thinkrelevance.com/blog/2013/06/04/clojure-workflow-reloaded中的想法实施到我的代码库中。
我有一个 dao 层,我现在需要传入一个数据库以避免全局状态。让我失望的一件事是这句话:
任何需要这些组件之一的函数都必须将其作为参数。这并不像看起来那么繁琐:每个函数最多有一个额外的参数来提供它运行的“上下文”。该上下文可能是整个系统对象,但更多时候是某个子集。通过明智地使用词法闭包,多余的参数会从大多数代码中消失。
我应该在哪里使用闭包以避免每次调用都传递全局状态?一个示例是在 dao 层中创建一个 init 函数,如下所示:
(defprotocol Persistable
(collection-name [this]))
(def save nil)
(defn init [{:keys [db]}]
(alter-var-root #'save (fn [_] (fn [obj] (mc/insert-and-return db (collection-name obj) obj WriteConcern/SAFE)))))
这样我可以像这样从系统/启动函数启动我的 dao 层:
(defn start
[{:keys [db] :as system}]
(let [d (-> db
(mc/connect)
(mc/get-db "my-test"))]
(dao/init d)
(assoc system :db d)))
这行得通,但感觉有点恶心。有没有更好的办法?如果可能的话,我想避免强制我的 dao 层的客户端每次使用函数时都必须传递数据库。