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

欢迎!请查看关于页面以了解更多关于该网站的信息。

0 投票
错误
当*compile-path*不是一个字符串而是一个java.io.File(目前boot-clj就是这样做的)。当你尝试在boot repl中使用它时,编译器会抛出类型转换异常。常见的建议是不从repl编译,人们说不要使用gen-class。但我觉得当你确实需要使用gen-class并快速迭代时,这是件不幸的事。

我没有对所有旧版本进行测试,但git blame显示,帮助解决这里的代码这10年内没有改变
https://github.com/clojure/clojure/blame/master/src/clj/clojure/core.clj#L6073
https://github.com/clojure/clojure/blame/master/src/jvm/clojure/lang/Compiler.java#L7643

我的解决方案现在是这样的


(defn compile-safely [ns-name]
  (let [compile-path-class *compile-path*
        corrected-path (cond
                         (instance? java.io.File compile-path-class) (.getAbsolutePath *compile-path*)
                         (instance? String compile-path-class)       *compile-path*
                         :else
                         (throw (ex-info "不支持的类" {:class (class compile-path-class)})))]
    (binding [*compile-path* corrected-path]
      (compile ns-name))))


我建议在'clojure.core/compile'函数中添加类型检查。

5 答案

0 投票
0 投票

由 jeroen发表的评论:

以下在 boot-clj 中有效(因为 Leiningen 和 clj cli 中的类路径问题):https://gist.github.com/jeroenvandijk/8187413d24433545eeb9579538a903f7#file-repl_compile-clj-L39-L52

0 投票

评论者为:alexmiller

你能提供一个可重复的例子吗?

看起来 ClassCastException 对错误类型非常准确的描述。

0 投票
_评论者为:jeroen

我已经尽可能简化了这个例子。你是对的,ClassCastException 完全准确,但你只有在弄清楚它与 *compile-path* 相关之后才会意识到这一点。这个错误报告主要关于管理期望(关于 Clojure 用户)。



clj -A:new app example.error
cd example.error
clj

 ;; clj 将 *compile-path* 正确设置为字符串(尽管不在类路径中)
 *compile-path* ;=> "classes"
 (let [f (clojure.java.io/file "classes/foo.clj")]
   (clojure.java.io/make-parents f)
   (spit f "(ns foo) (gen-class :name my.new.Class)")
   (compile 'foo)
   (clojure.java.shell/sh "ls" "classes"))
 ; 成功,尽管你仍然无法加载该类
 ; #=> {:exit 0, :out  "clojure\nfoo$fn__134.class\nfoo$fn__166.class\nfoo$fn__174.class\nfoo$loading__6549__auto____132.class\nfoo$loading__6549__auto____164.class\nfoo$loading__6549__auto____172.class\nfoo.clj\nfoo__init.class\n", :err ""}
 

 ;; 从 boot 工程项目
 ;; boot.user=> *compile-path*
 ;; #object[java.io.File 0x5266b1a "/var/folders/ck/9zqvs0zx5vsf2bc7rvv1pwc40000gn/T/boot-repl8170014881711480931"]
 ;; 这可以在 clj repl 中重现
 (binding [*compile-path* (clojure.java.io/file "classes")]
   (compile 'foo)
   )

 Syntax error compiling fn* at (foo.clj:1:1).
 原因:无法将 java.io.File 转换为 java.lang.String
 
 
0 投票
参考:https://clojure.atlassian.net/browse/CLJ-2412(由 jeroen 报告)
...