JS引擎(3):java项目里面的JS业务集成—Rhino/Nashorn
Author:zhoulujun Date:
初步整理,内容非常杂乱,待完善……
Mozilla Rhino是一个纯JavaScript语言的实现;而Node.js是一套JavaScript程序的运行时环境,不但包括由V8提供的JavaScript核心语言的实现,还包括丰富的库,最有特点的就是其基于事件的I/O库。用一句话描述Node.js就是其广告词之一:“Evented I/O for V8 JavaScript”。、
基于JVM的语言实现是服务器端程序的流行选择之一,而Rhino正是用Java实现、运行在JVM上的JavaScript引擎,可以无缝使用Java丰富的核心库和第三方库,所以有不少基于Rhino的服务器端JavaScript解决方案。
Rhino只是一个Java语言写JavaScript引擎,可以执行JavaScript代码而已。但是平时,我们部分逻辑业务,用rhino做,还是不成问题的
Java 和 JavaScript 交互
java与JavaScript交互,有两种引擎,Rhino、Nashorn
从根本上讲在 Java 项目中嵌入 JavaScript 脚本引擎,最重要的一点是实现 Java 和 JavaScript 之间的数据共享。Rhino为我们提供了强大的Java 和 JavaScript 交互的方法。
Rhino脚本引擎主要的类图
ScriptEngine接口就是脚本引擎,用于执行脚本计算结果的接口,其实现类是AbstractScriptEngine和底层的RhinoScriptEngine,这些类是脚本引擎的核心类。eval(String script)和eval(String script, Bindings n)两个方法就是执行一段脚本返回计算结果的两个方法,第二个方法会传入上下文,即运行脚本时,脚本需要使用上下文中设置的Java变量的方法或属性。
上下文是有范围的,分三种范围:
全局的:所有脚本引擎都可以使用的,由ScriptContext. GLOBAL_SCOPE常量来定义其范围的名称。ScriptEngine.getBindings(ScriptContext.GLOBAL_SCOPE)可以获GLOBAL取这类上下文。
引擎即的:只是一个脚本引擎可以使用的,由ScriptContext. ENGINE_SCOPE常量来定义其范围的名称。ScriptEngine.getBindings(ScriptContext.ENGINE_SCOPE)可以获取这类上下文。
局部的:引擎的一次计算用到的Bindings,没有常量定义,ScriptEngine.createBindings()创建的就是这类上下文。
Java中使用JavaScript脚本-Nashorn
创建JavaScript容器用户存储脚本 ScirptContainer.java
public class ScirptContainer { public static ScriptEngine engine;//脚本引擎 static { ScriptEngineManager manager = new ScriptEngineManager();//脚本引擎管理 engine = manager.getEngineByName("nashorn");//获取nashorn脚本引擎 engine.getContext().getWriter();//获取正文并且写入 } private ConcurrentHashMap<Integer, CompiledScript> scripts = new ConcurrentHashMap<>();//脚本存储容器 public CompiledScript getCompiledScript(String script) throws ScriptException{ //判断脚本是否为空 if(script == null || "".equals(script)){ throw new ScriptException("JavaScript empty"); } //获取脚本Hash int hashCode = script.hashCode(); //从容器中获取脚本 CompiledScript compiledScript = scripts.get(hashCode); if(compiledScript == null){ //容器中无脚本创建脚本对象 Compilable compilable = (Compilable) engine; //编译JavaScript脚本 compiledScript = compilable.compile(script); //脚本对象存入容器中 scripts.put(hashCode, compiledScript); } return compiledScript; } }
Java执行JavaScript脚本
public class ScriptHandler { //创建容器对象 private ScirptContainer scirptContainer = new ScirptContainer(); //需要执行的对象 String js1 = "function scriptFunction(obj){ var a = 1; var b = 2; return obj.sum(a,b); } scriptFunction(obj);"; @Test public void test() throws ScriptException{ //获取脚本对象 CompiledScript c1 = scirptContainer.getCompiledScript(js1); //创建参数绑定 Bindings bindings = scirptContainer.engine.createBindings(); //obj参数绑定SumTest类 bindings.put("obj", new SumTest()); //执行JavaScript脚本并且打印返回值 System.out.println(c1.eval(bindings)); } }
注意事项:
脚本中scriptFunction(obj);是必须存在,否则不会执行方法.
脚本中可以创建Java对象,需要全类名如var map = new java.util.HashMap();
在 Java 中获取 JavaScript 脚本中的变量
可以用于获取脚本运行过程中的一些临时信息变量
Context ct = Context.enter(); Scriptable scope = ct.initStandardObjects(); ct.evaluateString(scope, "var test = 'Successful';", null, 1, null); Object jsObject = scope.get("test" , scope); if (jsObject == Scriptable.NOT_FOUND) { System.out.println("test is not defined."); } else { System.out.println("test is " + Context.toString(jsObject)); }
在上述的示例代码中,我们同样用到了 scope 变量,这个是 JavaScript 运行时的全局变量,你可以将它理解成为一个容器,里面包含了 JavaScript 运行过程中的所有信息,所以如果你希望取得 JavaScript 过程中的某些信息,应首先考虑该对象。
在 Java 中调用 JavaScript 脚本中的函数
在前文中介绍了如何在 JavaScript 脚本运行Java函数,同样我们也可以在Java代码中运行JavaScript中的函数,示例代码如下:
Context ct = Context.enter(); Scriptable scope = ct.initStandardObjects(); ct.evaluateString(scope,"function test(name){return 'Successful!'+name;}", null, 1, null); Object functionObject = scope.get("test" , scope); if (!(functionObject instanceof Function)) { System.out.println("test is undefined or not a function."); } else { Object testArgs[] = {"Ceven"}; Function test = (Function)functionObject; Object result = test.call(ct, scope, scope, testArgs); System.out.println(Context.toString(result)); }
JavaScript中使用Java的功能
将一部分功能用 Java 实现,并在JavaScript中调用,可以方便解决java转换到JavaScript产生的个别问题,示例代码如下:
Context ct = Context.enter(); Scriptable scope = ct.initStandardObjects(); String str = "var test={};"; str += "test.call=function(){return 'Successful!';};"; str += "java.lang.System.out.println(test.call())"; ct.evaluateString(scope, str, null, 1, null);
可以在JavaScript使用Rhino提供的importPackage和importClass,详见文末链接
将Java对象共享到JavaScript
可以将 Java 对象转换为 JavaScript 对象,并用putProperty方法添加到运行环境中,示例代码如下:
Context ct = Context.enter(); Scriptable scope = ct.initStandardObjects(); Object out = Context.javaToJS(System.out, scope); ScriptableObject.putProperty(scope, "out", out); ct.evaluateString(scope, "out.println('Successful!')", null, 1, null);
关于Java跑Rhino的文章:
https://www.tutorialspoint.com/java8/java8_nashorn_java_script.htm
https://stackoverflow.com/questions/22502630/switching-from-rhino-to-nashorn
https://blogs.oracle.com/java-platform-group/nashorn,-the-rhino-in-the-room
Rhino -- 基于java的javascript实现 https://www.cnblogs.com/cczw/archive/2012/07/16/2593957.html
Rhino初试 https://www.jianshu.com/p/689edaa07a98
Java中使用JavaScript脚本 https://www.debug8.com/java/t_40597.html
转载本站文章《JS引擎(3):java项目里面的JS业务集成—Rhino/Nashorn》,
请注明出处:https://www.zhoulujun.cn/html/webfront/browser/webkit/2020_0719_8525.html