我正在尝试编写一个Java API,它将从在JDK 7 JS引擎下运行的服务器端JavaScript调用。这个想法是让消费者能够编写如下所示的JS代码,注册一个稍后由另一个Java方法调用执行的回调:
myJavaObj.registerCallback(function (param) {
// do stuff here
});
myJavaObj.methodThatTriggersCallback();
以下是我正在使用的一些测试代码:
import javax.script.Invocable;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;
public class JSTestClient {
private String successCallback;
public void handleSuccess(String successCallback) {
this.successCallback = successCallback;
}
public void doStuff() throws ScriptException, NoSuchMethodException {
ScriptEngineManager manager = new ScriptEngineManager();
Invocable engine = (Invocable) manager.getEngineByName("JavaScript");
engine.invokeFunction(successCallback, "TEST SUCCESS");
}
}
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;
public class Main {
public static void main(String[] args) {
ScriptEngineManager manager = new ScriptEngineManager();
ScriptEngine engine = manager.getEngineByName("JavaScript");
String js =
"var client = new Packages.JSTestClient();\n" +
"client.handleSuccess(function (response) {\n" +
" java.lang.System.out.println(response);\n" +
"});\n" +
"client.doStuff();";
try {
engine.eval(js); // Expecting this to output "TEST SUCCESS"
} catch (ScriptException e) {
e.printStackTrace();
}
}
}
但是当我运行它时,我得到一个java.lang.NoSuchMethodException,因为它正在解释字符串:
function (response) { java.lang.System.out.println(response); }作为函数名。有没有办法创建这样的回调,还是需要使用其他约定?
答案 0 :(得分:1)
您的handleSuccess
方法会以String
为参数,但您可以使用JavaScript function
对象调用它。您需要将其更改为接受Function
。
由于您正在使用Java 7,因此您使用的是已修改的Mozilla Rhino引擎,其中实现类已从包org.mozilla.javascript
移至sun.org.mozilla.javascript.internal
。
要使代码运行,请将JSTestClient
替换为:
import sun.org.mozilla.javascript.internal.Context;
import sun.org.mozilla.javascript.internal.Function;
public class JSTestClient {
private Function successCallback;
public void handleSuccess(Function successCallback) {
this.successCallback = successCallback;
}
public void doStuff() {
this.successCallback.call(Context.getCurrentContext(), null, null,
new Object[] { "TEST SUCCESS" });
}
}
请注意,Java 8已更改为Nashorn引擎,与Nashorn交互的代码完全不同。