Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 669|回复: 0

在Java中使用脚本语言

[复制链接]

该用户从未签到

发表于 2011-9-13 21:37:57 | 显示全部楼层 |阅读模式
有一段时间没有搞java,对于Java5以来一些新特性了解也不多,这几天看Solr的DIH,发现个很不错的配置支持–脚本引擎。以前提Java,是一处编译到处运行,现在可以说是一个平台多种语言。借此机会,整理了下Java6中引入的脚本引擎的相关特点和功能。
1、可用的脚本引擎
Java6提供对执行脚本语言的支持,这个支持来自于JSR223规范,对应的包是javax.script。默认情况下,Java6只支持 JavaScript脚本,它底层的实现是Mozilla Rhino(Rhino意为犀牛,是不是想起那本JavaScript大部头的封面了?),它是个纯Java的JavaScript实现。可以通过下面的代码列出当前环境中支持的脚本引擎:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ListScriptEngineFactory> factories = manager.getEngineFactories();        for (ScriptEngineFactory f : factories) {            System.out.println(                    "egine name:"+f.getEngineName()+                    ",engine version:"+f.getEngineVersion()+                    ",language name:"+f.getLanguageName()+                    ",language version:"+f.getLanguageVersion()+                    ",names:"+f.getNames()+                    ",mime:"+f.getMimeTypes()+                    ",extension:"+f.getExtensions());        }[/pre]

在我的机器上的输出是:egine name:Mozilla Rhino,engine version:1.6 release 2,language name:ECMAScript,language version:1.6,names:[js, rhino, JavaScript, javascript, ECMAScript, ecmascript],mime:[application/javascript, application/ecmascript, text/javascript, text/ecmascript],extension:[js]。可以看到,Java内置只支持JavaScript一种脚本。但是,只要遵循 JSR223,便可以扩展支持多种脚本语言,可以从https://scripting.dev.java.net/上查找当前已被支持的脚本的第三方库。
2、hello script
接下来给出在Java中使用JavaScript的Hello world示例:
[pre]        ScriptEngineManager manager = new ScriptEngineManager ();        ScriptEngine engine = manager.getEngineByName ("js");        String script = "print ('hello script')";        try {            engine.eval (script);        } catch (ScriptException e) {            e.printStackTrace();        }[/pre]

使用的API还是很简单的,ScriptEngineManager是ScriptEngine的工厂,实例化该工厂的时候会加载可用的所有脚本引擎。从工厂中创建ScriptEngine可以使用getEngineByName、getEngineByExtension或 getEngineByMimeType来得到,只要参数名字能对上。执行脚本调用eval方法即可(效果等同于JavaScript中的eval)。
3、传递变量
可以向脚本中传递变量,使得Java代码可以和脚本代码交互,示例如下:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ScriptEngine engine = manager.getEngineByName("js");        engine.put("a", 4);        engine.put("b", 6);        try {            Object maxNum = engine.eval("function max_num(a,b){return (a>b)?a:b;}max_num(a,b);");            System.out.println("max_num:" + maxNum);        } catch (Exception e) {            e.printStackTrace();        }[/pre]

输出内容:max_num:6
对于上面put的变量,它作用于自身engine范围内,也就是ScriptContext.ENGINE_SCOPE,put 的变量放到一个叫Bindings的Map中,可以通过 engine.getBindings(ScriptContext.ENGINE_SCOPE).get(“a”);得到put的内容。和ENGINE_SCOPE相对,还有个ScriptContext.GLOBAL_SCOPE 作用域,其作用的变量是由同一ScriptEngineFactory创建的所有ScriptEngine共享的全局作用域。
4、动态调用
上面的例子中定义了一个JavaScript函数max_num,可以通过Invocable接口来多次调用脚本库中的函数,Invocable接口是 ScriptEngine可选实现的接口。下面是个使用示例:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ScriptEngine engine = manager.getEngineByName("js");        try {            engine.eval("function max_num(a,b){return (a>b)?a:b;}");            Invocable invoke = (Invocable) engine;            Object maxNum = invoke.invokeFunction("max_num",4,6);            System.out.println(maxNum);            maxNum = invoke.invokeFunction("max_num", 7,6);            System.out.println(maxNum);        } catch (Exception e) {            // TODO: handle exception        }[/pre]

上面的invokeFunction,第一个参数调用的脚本函数名,后面跟的可变参数是对应的脚本函数参数。
Invocable还有个很酷的功能,就是动态实现接口,它可以从脚本引擎中得到Java Interface 的实例;也就是说,可以定义个一个Java接口,其实现是由脚本完成。以上面的例子为例,定义接口JSLib,该接口中的函数和JavaScript中的函数签名保持一致:
[pre]    public interface JSLib {        public int max_num(int a,int b);    }[/pre]

调用示例:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ScriptEngine engine = manager.getEngineByName("js");        try {            engine.eval("function max_num(a,b){return (a>b)?a:b;}");            Invocable invoke = (Invocable) engine;            JSLib jslib = invoke.getInterface(JSLib.class);            int maxNum = jslib.max_num(4,6);            System.out.println(maxNum);        } catch (Exception e) {            // TODO: handle exception        }[/pre]

5、使用 Java 对象
可以在JavaScript中使用Java代码,这确实是很酷的事情。在Rhino中,可以通过importClass导入一个类,也可以通过importPackage导入一个包,也可以直接使用全路经的类。在创建对象时,new也不是必须的。示例代码如下:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ScriptEngine engine = manager.getEngineByName("js");        try {            String script = "var list = java.util.ArrayList();list.add(\"Kafka0102\");print(list.get(0));";            engine.eval(script);        } catch (Exception e) {            e.printStackTrace();        }[/pre]

6、编译执行
脚本引擎默认是解释执行的,如果需要反复执行脚本,可以使用它的可选接口Compilable来编译执行脚本,以获得更好的性能,示例代码如下:
[pre]        ScriptEngineManager manager = new ScriptEngineManager();        ScriptEngine engine = manager.getEngineByName("js");        try {            Compilable compEngine = (Compilable) engine;            CompiledScript script = compEngine.compile("function max_num(a,b){return (a>b)?a:b;}");            script.eval();            Invocable invoke = (Invocable) engine;            Object maxNum = invoke.invokeFunction("max_num",4,6);            System.out.println(maxNum);        } catch (Exception e) {            e.printStackTrace();        }[/pre]

7、总结
除了上面提到的特性,脚本引擎还有一些不错的功能,比如可以执行脚本文件,可以由多线程异步执行脚本等功能。引入脚本引擎,可以对一些配置扩展和业务规则做更强大而灵活的支持,也方便使用者选择自己熟悉的脚本语言来编写业务规则等。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

GMT+8, 2024-5-7 07:55 , Processed in 0.379740 second(s), 49 queries .

Powered by Discuz! X3.4

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表