当前位置: 首页 > 知识库问答 >
问题:

Nashorn CommandListener$$NashornJavaAdapter无法强制转换到CommandListener

曹光霁
2023-03-14

我使用Nashorn与Java制作一个模块化插件系统。插件将使用Javascript编写。一个重要的特性是具有可以用Javascript代码编写的处理程序。一开始,我的目标是制作一个简单的命令系统。javascript实现一个接口,并调用Java方法来注册命令。然而,我得到一个错误。我确保我使用的是Nashorn(var usingNashorn=typeof进口类!=="函数";返回true。

Javascript:

var CommandListener = Java.extend(Java.type("com.techsdev.scriptengine.listeners.CommandListener"), {
    invoke : function(sender, command, args) {
        java.lang.System.out.println("Received a command: " + command);
    }
});

var listen = function(scriptManager) {
    var listener = new CommandListener();
    scriptManager.registerCommand("plugin name", "test", listener);
}

Java代码:调用“侦听”:

try {
    engine.eval(new FileReader(f));
    Invocable invocable = (Invocable) engine;
    invocable.invokeFunction("listen", this);
} catch(Exception e) {
    logger.error("Failed to load script "+f.getName(), e);
}

其中'f'是Javascript文件,其中'Engine'是Nashorn ScriptEngine,其中'this'是ScriptManager

在ScriptManager类中,此方法负责注册实际命令:

public void registerCommand(String plugin, String command, CommandListener listener) {
        if(commandHandlers.containsKey(command.toLowerCase())) {
            logger.warn("Command "+command+" tried to be registered, but is already registered!");
            return;
        }

        commandHandlers.put(command.toLowerCase(), listener);
}

但是,此代码引发以下异常:

java.lang.ClassCastException: com.techsdev.scriptengine.listeners.CommandListener$$NashornJavaAdapter cannot be cast to com.techsdev.scriptengine.listeners.CommandListener
    at com.techsdev.scriptengine.JsScriptManager.registerCommand(JsScriptManager.java:168) ~[scriptengine-mod.jar:?]
    at jdk.nashorn.internal.scripts.Script$Recompilation$2$616A$\^eval\_.listen(<eval>:22) ~[?:?]
    at jdk.nashorn.internal.runtime.ScriptFunctionData.invoke(ScriptFunctionData.java:625) ~[nashorn.jar:?]
    at jdk.nashorn.internal.runtime.ScriptFunction.invoke(ScriptFunction.java:494) ~[nashorn.jar:?]
    at jdk.nashorn.internal.runtime.ScriptRuntime.apply(ScriptRuntime.java:393) ~[nashorn.jar:?]
    at jdk.nashorn.api.scripting.ScriptObjectMirror.callMember(ScriptObjectMirror.java:199) ~[nashorn.jar:?]
    at jdk.nashorn.api.scripting.NashornScriptEngine.invokeImpl(NashornScriptEngine.java:383) ~[nashorn.jar:?]
    at jdk.nashorn.api.scripting.NashornScriptEngine.invokeFunction(NashornScriptEngine.java:190) ~[nashorn.jar:?]
    at com.techsdev.scriptengine.JsScriptManager.loadFile(JsScriptManager.java:134) [JsScriptManager.class:?]
    at com.techsdev.scriptengine.JsScriptManager.loadFolder(JsScriptManager.java:116) [JsScriptManager.class:?]
    at com.techsdev.scriptengine.JsScriptManager.init(JsScriptManager.java:104) [JsScriptManager.class:?]

如果我遗漏了什么,请告诉我。先谢谢你。

共有1个答案

万俟英锐
2023-03-14

我会打印com的类加载器。techsdev。脚本引擎。听众。CommandListener来自脚本和Java代码。

从JavaScript:

print(Java.type("com.techsdev.scriptengine.listeners.CommandListener").class.classLoader)

来自Java:

System.out.println(com.techsdev.scriptengine.listeners.CommandListener.class.getClassLoader());

如果两个不同的加载程序加载相同的(完全限定的)命名类字节,则从JVM的角度来看,这些(运行时)类是不同的。如果您从javascript和java代码中看到不同的类加载器,则很可能存在类路径/类加载器问题。

 类似资料: