我如何编写一个Java方法,从服务器端JavaScript调用,将JS回调函数作为参数?

时间:2018-02-22 03:22:40

标签: javascript java rhino

我正在尝试编写一个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);
}
作为函数名。有没有办法创建这样的回调,还是需要使用其他约定?

1 个答案:

答案 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交互的代码完全不同。