2017-04-12 104 views
1

Java的子类调用函数我有一个JavaScript脚本,看起来像这样:在犀牛

function run(database) { 
    var result = database.query("query", "some resource name"); 
    //operations on result 
    return result; 
} 

,我有Java代码执行的脚本是这样的:

public Object execute(String script, Database database) { 
    NashornScriptEngineFactory nsef = new NashornScriptEngineFactory(); 
    ScriptEngine engine = nsef.getScriptEngine(); 
    try { 
     engine.eval(script); 
     Invocable invocable = (Invocable) engine; 
     return invocable.invokeFunction("run", database); 
    } catch(ScriptException e) { 
     throw new RuntimeException(e); 
    } 
} 

Database是一个包含几个方法定义的接口,但是而不是包含query方法。我打电话execute,执行Database,称之为DatabaseImpl确实query方法。这将是多态的,并且该脚本需要知道传递给它的Database实例上有哪些方法可用。我决定不要使用这种泛型,因为它们在运行时被擦除,所以JavaScript无法使用它们,因此脚本编写者需要正确地使用这些类型。

然而,当运行此代码时,得到以下异常:

javax.script.ScriptException: TypeError: database.query is not a function in <eval> at line number 25 

基本上,要点是,我有它实现一个接口的对象,并调用该特定实例可实现的方法,但不是接口定义的一部分。我的印象是,这应该仍然有效,但事实并非如此。对我而言,我需要在脚本中进行子播以访问query方法(这甚至有可能吗?),为什么我会得到这个错误?是否因为该方法不能从接口定义中获得?有没有解决方法?

谢谢。

回答

1

这是主要的类:

package so; 
import java.io.InputStreamReader; 
import javax.script.Invocable; 
import javax.script.ScriptEngine; 
import javax.script.ScriptEngineManager; 

public class Nashorn { 
    public static void main(String[] args) { 
     try (InputStreamReader in = resource()) { 
      ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn"); 
      engine.eval(in); 
      Invocable invocable = (Invocable) engine; 
      Database database = new DatabaseImpl(); 
      Object x = invocable.invokeFunction("run", database); 
      System.out.println(x); 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 
    } 

    private static InputStreamReader resource() throws Exception { 
     return new InputStreamReader(Nashorn.class.getResourceAsStream("db.js"), "utf-8"); 
    } 

} 

接口和实现

package so; 

public interface Database { 
    void connect(); 
} 

package so; 

import java.util.ArrayList; 
import java.util.Arrays; 
import java.util.List; 

public class DatabaseImpl implements Database { 

    @Override 
    public void connect() { 
     System.out.println("Connecting"); 
    } 

    public List<?> query(String ... stmt){ 
     List<String> lst = new ArrayList<>(); 
     lst.addAll(Arrays.asList(stmt)); 
     lst.addAll(Arrays.asList("A","B","C")); 
     return lst; 
    } 

} 

JavaScript文件(以便/ db.js)

function run(database) { 
    var result = database.query("query", "some resource name"); 
    //operations on result 
    return result; 
} 

运行结果在:

[query, some resource name, A, B, C] 

它基本上工作。

+0

可能PEBCAC然后...我会继续寻找。感谢您检查了这一点。 – Peter