Clojure符号评估错误

时间:2012-01-15 10:00:13

标签: clojure read-eval-print-loop leiningen uberjar clojure-1.3

所以我目前有这段代码:

(ns contact-form.core
  (:gen-class))

(def foo "Hello World!")

(defn some-func [a-symbol]
  (println (str a-symbol "'s value is: " (eval a-symbol))))

(defn -main [& args]
  (some-func 'foo))

在Emacs中执行C-c C-k之后,我得到以下输出:

contact-form.core> (-main)
foo's value is: Hello World!
nil

但是当我执行lein uberjar并运行生成的jar文件时,我收到错误:

Exception in thread "main" java.lang.RuntimeException: Unable to resolve symbol: foo in this context, compiling:(NO_SOURCE_PATH:0)
    at clojure.lang.Compiler.analyze(Compiler.java:6235)
    at clojure.lang.Compiler.analyze(Compiler.java:6177)
    at clojure.lang.Compiler.eval(Compiler.java:6469)
    at clojure.lang.Compiler.eval(Compiler.java:6431)
    at clojure.core$eval.invoke(core.clj:2795)
    at contact_form.core$some_func.invoke(core.clj:7)
    at contact_form.core$_main.doInvoke(core.clj:10)
    at clojure.lang.RestFn.invoke(RestFn.java:397)
    at clojure.lang.AFn.applyToHelper(AFn.java:159)
    at clojure.lang.RestFn.applyTo(RestFn.java:132)
    at contact_form.core.main(Unknown Source)
Caused by: java.lang.RuntimeException: Unable to resolve symbol: foo in this context
    at clojure.lang.Util.runtimeException(Util.java:156)
    at clojure.lang.Compiler.resolveIn(Compiler.java:6720)
    at clojure.lang.Compiler.resolve(Compiler.java:6664)
    at clojure.lang.Compiler.analyzeSymbol(Compiler.java:6625)
    at clojure.lang.Compiler.analyze(Compiler.java:6198)
    ... 10 more

所以我有两个问题:

  1. 为什么uberjar的功能与REPL完全不同?
  2. 我该怎么做才能解决这个问题?

1 个答案:

答案 0 :(得分:5)

<强> 1。为什么uberjar的功能与REPL不同?

错误“NO_SOURCE_PATH”的原因是您目前不在定义'foo的命名空间中。

为了说明:如果我在我的REPL中评估你的代码并执行它,它会把我放在contact-form.core命名空间中,因为你期望因为(ns contact-form.core)由REPL评估,但是如果我切换到user命名空间并调用-main我可以产生相同的错误:

contact-form.core=> (-main)
foo's value is: Hello World!
nil
contact-form.core=> (ns user)
nil
user=> (contact-form.core/-main)
CompilerException java.lang.RuntimeException: Unable to resolve symbol: foo in this context, compiling:(NO_SOURCE_PATH:120) 
user=> 

因此,通过为独立的uberjar执行(在REPL之外)指定main的入口点,它等同于从jar中的默认命名空间调用(contact-form.core/-main) clojure.core ,因为(ns contact-form.core)尚未评估。结果:main可以使用函数的完全限定(命名空间)路径执行,但contact-form.core中没有符号在当前默认命名空间中可用。

<强> 2。修复

解决方案是首先明确切换到您的命名空间。

(defn -main [& args]
  (use 'contact-form.core)
  (some-func 'foo))