10

基本上我试图将一个javascript函数传递给一个Java方法来作为脚本的回调。

我可以做到——有点——但我收到的对象是 sun.org.mozilla.javascript.internal.InterpretedFunction,我看不到调用它的方法。

有任何想法吗?

这是我到目前为止所拥有的:

var someNumber = 0;

function start() {
   // log is just an log4j instance added to the Bindings
   log.info("started....");
   someNumber = 20;

    // Test is a unit test object with this method on it (taking Object as a param).
    test.callFromRhino(junk);
}

function junk() {
    log.info("called back " + someNumber);
}
4

4 回答 4

10

实现一个接口:

import javax.script.*;

public class CallBack {
  public void invoke(Runnable runnable) {
    runnable.run();
  }

  public static void main(String[] args) throws ScriptException {
    ScriptEngine js = new ScriptEngineManager().getEngineByExtension("js");
    js.getContext().setAttribute("callBack", new CallBack(),
        ScriptContext.ENGINE_SCOPE);
    js.eval("var impl = { run: function () { print('Hello, World!'); } };\n"
        + "var runnable = new java.lang.Runnable(impl);\n"
        + "callBack.invoke(runnable);\n");
  }
}
于 2010-05-16T08:52:22.833 回答
7

sun.org.mozilla.javascript.internal.InterpretedFunction实现接口sun.org.mozilla.javascript.Function。该接口上有一个名为的方法,该方法call需要:

  • 一种Context
  • aScriptable用作范围
  • aScriptable用作this函数内的值
  • 一个数组Objects是函数的参数

所以,我的建议是在 java 中你将传递的对象转换为 asun.org.mozilla.javascript.Function和 call call。前两个参数可以是您最初从 java 中用于启动脚本的任何参数。你在那里使用它的方式,最后两个参数可以是nulland new Object[0]

于 2010-05-15T23:41:15.143 回答
2

解决方案实际上是在另一个脚本中调用它。这类作品:

import javax.script.*;

public class CallFunction {

    /**
     * @param args
     * @throws Exception oops!
     */
    public static void main(String[] args) throws Exception {
        ScriptEngine js = new ScriptEngineManager().getEngineByExtension("js");
        js.getContext().setAttribute("out", System.out, ScriptContext.ENGINE_SCOPE);
        Object a = js.eval(
                "out.println('Defining function a...');" +
                "function a() {out.println('hello from JavaScript!'); }" +
                "function foobar() {out.println('in foobar() definition');}" +    
                "out.println('Done!.');"
        );

        System.out.println(js.get("a")); // InterpretedFunction
        SimpleBindings bindings = new SimpleBindings();
        bindings.put("foobar",js.get("a"));
        js.eval("foobar();", bindings); // hello from JavaScript
        js.eval("foobar();"); // in foobar() definition
    }
}

当您取回对函数的引用时,您需要让引擎为您执行该函数。虽然不是很漂亮,但要求 js 使用一组特定的绑定为您 eval() 它实际上会为您完成这项工作。您需要注意您正在操作的变量属于正确的范围;我想这里很容易出错。

于 2010-10-20T19:09:35.010 回答
1

这个例子涵盖了用 javascript 实现 java 接口。这也可用于从 java 调用 javascript 回调。



package com.hal.research;

import javax.script.*;

public class CallFunction {
    /**
     * define contract for the callback 
     */
    static interface WhatEverYouWant {
        public String testMe(String a, String b);
    }
    /**
     * @param args
     */
    public static void main(String[] args) throws Exception {
        final ScriptEngineManager scriptManager = new ScriptEngineManager();
        final ScriptEngine js = scriptManager.getEngineByExtension("js");
        js.put("producer", new Object() {
            /**
             * @param call is a callback to be invoked
             */
            public void doSomethingWithIt(WhatEverYouWant call) {
                System.out.println("invoke callback javascript...");
                String result = call.testMe("a", "b");
                // do something with the result ...
                System.out.println("invoke callback...done, result: "+result);
            }
        });
        js.eval(  "var handler = {\"testMe\": function (a,b){return a + \" is concatenated to \"+ b;}};\n"
                + "var callback = new Packages.com.hal.research.CallFunction.WhatEverYouWant(handler);\n"
                + "producer.doSomethingWithIt(callback); ");
    }
}


于 2013-09-08T22:10:01.840 回答