5

编辑:在 Bukkit 论坛Github上还有一些与此问题相关的其他讨论。

所以,我知道一两个人没有运气尝试过这个。但我想我快到了。

一个问题:我不懂Java,所以这对我来说有点陌生。反正..

于是,我在 Clojure 中做了一个简单的类,如下:

(ns com.gdude2002.ClojurePlugin.mainclj
  (:gen-class
   :name com.gdude2002.ClojurePlugin.mainclj
   :extends org.bukkit.plugin.java.JavaPlugin)
  (:import org.bukkit.plugin.java.JavaPlugin))

(defn -onEnable [this] (java.util.logging.Logger/getLogger "Loaded clojure plugin!"))
(defn -onDisable [this] (java.util.logging.Logger/getLogger "Unloaded clojure plugin!"))

我使用clojure的compile函数把它编译成Java类,如下..

(set! *compile-path* ".")
(compile 'com.gdude2002.ClojurePlugin.mainclj)

然后我将它手动放入 jar 中,在 com/gdude2002/ClojurePlugin/mainclj.class 下(以及将 plugin.yml 放入根目录)。

到目前为止,一切都很好。这种方法不会让 bukkit 对我的代码(特别是)感到讨厌,我认为这是一件好事。现在,关于问题。

当我尝试使用这个手工制作的 jar 启动服务器时,我得到以下输出。

21:43:30 [SEVERE] Could not load 'plugins\plugin.jar' in folder 'plugins'
org.bukkit.plugin.InvalidPluginException: java.lang.NoClassDefFoundError: clojure/lang/IFn

“所以,”我想,“这看起来很简单——它找不到 Clojure,对吧?” 所以,我把clojure jar放在各种地方,同样的错误。我还将 META-INF/MANIFEST.MF 添加到 jar 中,其中包含Class-Path: ../lib. 依然没有。

所以,认为我很聪明,我将 clojure 文件夹从 Clojure jar 中拉出并推入我的,还将文件夹放在 ../lib 和其他我认为可能有帮助的地方。

现在,我得到这个错误..

21:51:33 [SEVERE] Could not load 'plugins\plugin.jar' in folder 'plugins'
org.bukkit.plugin.InvalidPluginException: java.lang.ExceptionInInitializerError
...
Caused by: java.io.FileNotFoundException: Could not locate clojure/core__init.class or clojure/core.clj on classpath:
...

问题是,就我所见,这两种东西都存在。

编辑:决定在这里显示反编译的类代码,以防万一。

// IntelliJ API Decompiler stub source generated from a class file
// Implementation of methods is not available

package com.gdude2002.ClojurePlugin;

public class mainclj extends org.bukkit.plugin.java.JavaPlugin {
    private static final clojure.lang.Var main__var;
    private static final clojure.lang.Var onEnable__var;
    private static final clojure.lang.Var getResource__var;
    private static final clojure.lang.Var onLoad__var;
    private static final clojure.lang.Var getLogger__var;
    private static final clojure.lang.Var saveDefaultConfig__var;
    private static final clojure.lang.Var getDescription__var;
    private static final clojure.lang.Var removeDDL__var;
    private static final clojure.lang.Var onDisable__var;
    private static final clojure.lang.Var isInitialized__var;
    private static final clojure.lang.Var saveResource__var;
    private static final clojure.lang.Var onCommand__var;
    private static final clojure.lang.Var getDefaultWorldGenerator__var;
    private static final clojure.lang.Var toString__var;
    private static final clojure.lang.Var getDataFolder__var;
    private static final clojure.lang.Var installDDL__var;
    private static final clojure.lang.Var getDatabase__var;
    private static final clojure.lang.Var getFile__var;
    private static final clojure.lang.Var getClassLoader__var;
    private static final clojure.lang.Var getCommand__var;
    private static final clojure.lang.Var getDatabaseClasses__var;
    private static final clojure.lang.Var getConfig__var;
    private static final clojure.lang.Var reloadConfig__var;
    private static final clojure.lang.Var clone__var;
    private static final clojure.lang.Var setEnabled__var;
    private static final clojure.lang.Var saveConfig__var;

    public mainclj() { /* compiled code */ }

    public java.io.File getDataFolder() { /* compiled code */ }

    public boolean onCommand(org.bukkit.command.CommandSender p0, org.bukkit.command.Command p1, java.lang.String p2, java.lang.String[] p3) { /* compiled code */ }

    public void reloadConfig() { /* compiled code */ }

    public org.bukkit.configuration.file.FileConfiguration getConfig() { /* compiled code */ }

    public java.io.File getFile() { /* compiled code */ }

    public void saveConfig() { /* compiled code */ }

    public org.bukkit.command.PluginCommand getCommand(java.lang.String p0) { /* compiled code */ }

    public void onEnable() { /* compiled code */ }

    public java.util.logging.Logger getLogger() { /* compiled code */ }

    public void onLoad() { /* compiled code */ }

    public java.lang.ClassLoader getClassLoader() { /* compiled code */ }

    public void saveDefaultConfig() { /* compiled code */ }

    public org.bukkit.plugin.PluginDescriptionFile getDescription() { /* compiled code */ }

    public com.avaje.ebean.EbeanServer getDatabase() { /* compiled code */ }

    public void removeDDL() { /* compiled code */ }

    public void onDisable() { /* compiled code */ }

    public boolean isInitialized() { /* compiled code */ }

    public org.bukkit.generator.ChunkGenerator getDefaultWorldGenerator(java.lang.String p0, java.lang.String p1) { /* compiled code */ }

    public void installDDL() { /* compiled code */ }

    public void saveResource(java.lang.String p0, boolean p1) { /* compiled code */ }

    public java.util.List getDatabaseClasses() { /* compiled code */ }

    public java.lang.String toString() { /* compiled code */ }

    public java.lang.Object clone() { /* compiled code */ }

    public void setEnabled(boolean p0) { /* compiled code */ }

    public java.io.InputStream getResource(java.lang.String p0) { /* compiled code */ }

    public static void main(java.lang.String[] p0) { /* compiled code */ }
}

编辑:评论中提到我应该发布我的 project.clj 文件。问题是,我没有!我直接在 mainclj.clj 上运行 compile.clj,并手动创建一个 jarfile。

在这一点上,我完全被难住了。有人对此有任何想法吗?

4

1 回答 1

1

我通过配置调用代码的类加载器解决了这个问题,如本文所述。

我的情况可能与调用 Java 代码时有所不同。换句话说,我在从使用 gen-class (Clojure) 创建的类中调用方法之前,立即在我的 CommandExecutor (Java) 中调用了该代码。

于 2012-12-13T15:24:59.553 回答