从java调用Clojure时Clojure状态的范围

时间:2017-07-20 12:59:15

标签: clojure

我是Clojure的新手,所以也许这是一个非常明显的问题,但我找不到任何有用的答案。

我正在使用Spring Boot实现一个REST服务,它使用Clojure通过多次调用Compiler.eval()来计算结果。我有一些Clojure的东西应该在所有请求(常量,函数,全局变量等)之间共享,以及一些每个请求的东西应该被隔离并且每个请求都是唯一的。

我在类路径中使用clojure.jar将此服务作为可执行JAR运行。但是,从JVM的角度来看,你只能通过静态方法访问Clojure,这意味着只有一个Clojure实例"每个JVM。这意味着如果您在一次请求期间评估(def a 1)," a"其他请求也可见。

问题1:在Clojure中我应该在何处以及如何定义动态(请求范围的)变量/符号?

问题2:假设我想要一个完全"新的"每个请求的Clojure实例。我怎么能实现这一目标?那么,例如,我可以在同一个JVM中使用相同名称和相同名称空间但具有不同值的变量?我知道我可以使用一个新的类加载器加载一组全新的Clojure类,但这看起来非常麻烦。

2 个答案:

答案 0 :(得分:4)

你不应该在Clojure中使用全局变量,除非需要跟踪一些必须在所有函数之间共享的全局“状态”。它是一种功能性语言(大多数情况下),功能性的做事方式是将所有必需的东西作为参数传递。这就是你如何获得参考透明度以及如何避免函数看到和影响彼此的价值。

我不知道你为什么要使用SpringBoot,但也许你可以看看Clojure本地web库是如何工作的。它们通常会向处理程序传递某种“请求上下文”,其中存储了所有必要的信息。

例如YadaRing

所以回答你的(1):每次请求都传递它们。 并且(2):如果您为每个请求传递新值,则不需要它。

答案 1 :(得分:0)

虽然使用一个请求本地的全局变量(原谅丑陋的表达式)并不是Clojure中通常推荐的Web服务方式,但是假设你有一个有效的用例。一个这样的用例可能是建立一个像tryclj.com这样的在线REPL。

隔离请求的可能性是为每个请求创建一个临时(随机)命名空间,并在每次请求后擦除命名空间。 tryclojure does this for requests clojail使用sandbox {来自src/tryclojure/models/eval.clj的代码段:

(defn make-sandbox []
  (sandbox try-clojure-tester
           :timeout 2000
           :init '(do (require '[clojure.repl :refer [doc source]])
                      (future (Thread/sleep 600000)
                              (-> *ns* .getName remove-ns)))))