[英]leiningen: ClassNotFoundException for class present in project on “lein run” / “lein uberjar”
我正在嘗試實現我正在使用的遺留運行時所需的Java接口,並實例化此實現的實例以將其傳遞給運行時。 但是當我運行lein uberjar
我看到一個例外,那個類無法找到。 兩個名稱空間都在同一個leiningen項目中定義,所以我希望它們能夠看到彼此以及它們生成的類。
(ns man.core
(:require (man.gateway))
(:import (man ManGate)
(eu.m2machine.gw GatewayComponentFactory))
(:gen-class))
(defn -main [& args]
(let [starter (.starter (GatewayComponentFactory/get))
gateway (ManGate.)]
(.startup starter gateway)))
此代碼嘗試使用在同一項目中實現的類:
(ns man.gateway
(:import [eu.m2machine.gw.text GatewayIDFormatter])
(:gen-class
:name man.ManGate
:implements [eu.m2machine.gw.Gateway]
:prefix "gateway-"))
(defn gateway-startup [this])
(defn gateway-shutdown [this])
到目前為止,接口所需的兩個方法只有stub實現。 我可以在編譯代碼后得到他們的代碼。 定義接口(在作為依賴項添加的工件中):
package eu.m2machine.gw;
public interface Gateway {
void startup();
void shutdown();
}
我得到的例外是:
Exception in thread "main" java.lang.ClassNotFoundException: man.ManGate, compiling:(man/core.clj:1:1)
at clojure.lang.Compiler.load(Compiler.java:7391)
at clojure.lang.RT.loadResourceScript(RT.java:372)
at clojure.lang.RT.loadResourceScript(RT.java:363)
at clojure.lang.RT.load(RT.java:453)
at clojure.lang.RT.load(RT.java:419)
at clojure.core$load$fn__5677.invoke(core.clj:5893)
at clojure.core$load.invokeStatic(core.clj:5892)
at clojure.core$load.doInvoke(core.clj:5876)
at clojure.lang.RestFn.invoke(RestFn.java:408)
at clojure.core$load_one.invokeStatic(core.clj:5697)
at clojure.core$load_one.invoke(core.clj:5692)
at clojure.core$load_lib$fn__5626.invoke(core.clj:5737)
at clojure.core$load_lib.invokeStatic(core.clj:5736)
at clojure.core$load_lib.doInvoke(core.clj:5717)
at clojure.lang.RestFn.applyTo(RestFn.java:142)
at clojure.core$apply.invokeStatic(core.clj:648)
at clojure.core$load_libs.invokeStatic(core.clj:5774)
at clojure.core$load_libs.doInvoke(core.clj:5758)
at clojure.lang.RestFn.applyTo(RestFn.java:137)
at clojure.core$apply.invokeStatic(core.clj:648)
at clojure.core$require.invokeStatic(core.clj:5796)
at clojure.core$require.doInvoke(core.clj:5796)
at clojure.lang.RestFn.invoke(RestFn.java:408)
at user$eval5$fn__7.invoke(form-init2162986879369932757.clj:1)
at user$eval5.invokeStatic(form-init2162986879369932757.clj:1)
at user$eval5.invoke(form-init2162986879369932757.clj:1)
at clojure.lang.Compiler.eval(Compiler.java:6927)
at clojure.lang.Compiler.eval(Compiler.java:6917)
at clojure.lang.Compiler.load(Compiler.java:7379)
at clojure.lang.Compiler.loadFile(Compiler.java:7317)
at clojure.main$load_script.invokeStatic(main.clj:275)
at clojure.main$init_opt.invokeStatic(main.clj:277)
at clojure.main$init_opt.invoke(main.clj:277)
at clojure.main$initialize.invokeStatic(main.clj:308)
at clojure.main$null_opt.invokeStatic(main.clj:342)
at clojure.main$null_opt.invoke(main.clj:339)
at clojure.main$main.invokeStatic(main.clj:421)
at clojure.main$main.doInvoke(main.clj:384)
at clojure.lang.RestFn.invoke(RestFn.java:421)
at clojure.lang.Var.invoke(Var.java:383)
at clojure.lang.AFn.applyToHelper(AFn.java:156)
at clojure.lang.Var.applyTo(Var.java:700)
at clojure.main.main(main.java:37)
Caused by: java.lang.ClassNotFoundException: man.ManGate
at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
at clojure.lang.DynamicClassLoader.findClass(DynamicClassLoader.java:69)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at clojure.lang.DynamicClassLoader.loadClass(DynamicClassLoader.java:77)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
at java.lang.Class.forName0(Native Method)
at java.lang.Class.forName(Class.java:348)
at clojure.lang.RT.classForName(RT.java:2168)
at clojure.lang.RT.classForNameNonLoading(RT.java:2181)
at man.core$eval20$loading__5569__auto____21.invoke(core.clj:1)
at man.core$eval20.invokeStatic(core.clj:1)
at man.core$eval20.invoke(core.clj:1)
at clojure.lang.Compiler.eval(Compiler.java:6927)
at clojure.lang.Compiler.eval(Compiler.java:6916)
at clojure.lang.Compiler.load(Compiler.java:7379)
... 42 more
我想知道這是否可能是命名空間編譯順序的問題。 也許leiningen在man.core
之前處理man.ManGate
被創建了!?
編輯:
我可以通過改變實現界面的方式來解決我的問題:
(ns man.gateway
(:import [eu.m2machine.gw Gateway]))
(defn gateway []
(reify Gateway
(startup [this])
(shutdown [this]))
)
在命名空間man.core中,我替換了構造函數調用ManGate.
打電話給man.gateway/gateway
。
我仍然有興趣知道為什么上面的解決方案不起作用。
您的類在名稱空間man.gateway
定義,但它的完全限定類名是man.ManGate
。 更改FQCN到man.gateway.ManGate
兩個core.clj
和gateway.clj
應該解決的問題。
我花了很長時間來解決類似的問題,我在我的lein本地倉庫中有所需的罐子,但lein run
總是因ClassNotFoundException而失敗。 事實證明,即使我已經在我的project.clj
定義了所需的依賴項,我也錯過了require
在我的文件中使用確切的庫來引用庫方法。 所以,這是一個非常愚蠢的錯誤。但是,我在這里提出這個答案,以防萬一,它有助於一個Clojure新手。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.