java ScriptEngine 使用 (java运行脚本文件)
转自:http://www.tuicool.com/articles/imEbQbA
Java SE 6最引人注目的新功能之一就是内嵌了脚本支持。在默认情况下,Java SE 6只支持JavaScript,但这并不以为着Java SE 6只能支持JavaScript。在Java SE 6中提供了一些接口来定义一个脚本规范,也就是JSR223。通过实现这些接口,Java SE 6可以支持任意的脚本语言(如PHP或Ruby)。
运行第一个脚本程序
在使用Java SE 6运行脚本之前,必须要知道你的Java SE 6支持什么脚本语言。在javax.script包中有很多的类,但这些类中最主要的是ScriptEngineManager。可以通过这个类得到当前 Java SE 6所支持的所有脚本。如下面例子将列出所有可以使用的脚本引擎工厂。
- import javax.script.*;
- import java.io.*;
- import java.util.*;
- import static java.lang.System.*;
- public class ListScriptEngines
- {
- public static void main(String args[]){
- ScriptEngineManager manager = new ScriptEngineManager();
- // 得到所有的脚本引擎工厂
- List factories = manager.getEngineFactories();
- // 这是Java SE 5 和Java SE 6的新For语句语法
- for (ScriptEngineFactory factory: factories){
- // 打印脚本信息
- out.printf("Name: %s%n" +
- "Version: %s%n" +
- "Language name: %s%n" +
- "Language version: %s%n" +
- "Extensions: %s%n" +
- "Mime types: %s%n" +
- "Names: %s%n",
- factory.getEngineName(),
- factory.getEngineVersion(),
- factory.getLanguageName(),
- factory.getLanguageVersion(),
- factory.getExtensions(),
- factory.getMimeTypes(),
- factory.getNames());
- // 得到当前的脚本引擎
- ScriptEngine engine = factory.getScriptEngine();
- } } }
上面的例子必须要在Java SE 6中编译。其中import static java.lang.System.*是新的语法,将System中的所有静态成员进行引用,以后就可以直接使用out、in或err了。
通过运行java ListScriptEngines,将显示如下信息
- Name: Mozilla Rhino
- Version: 1.6 release 2
- Language name: ECMAScript
- Language version: 1.6
- Extensions: [js]
- Mime types: [application/javascript, application/ecmascript, text/javascript, text/ecmascript]
- Names: [js, rhino, JavaScript, javascript, ECMAScript, ecmascript]
在最下面一行是脚本的别名,也就是使用它们中的任意一个都可以。得到一个具体的脚本引擎有3种方法。
- 根据扩展名得到脚本引擎
- ScriptEngine engine = manager.getEngineByExtension("js");
getEngineByExtension的参数就是Extensions:[js]中[…]里的部分。
- 根据Mime类型得到脚本引擎
- ScriptEngine engine = manager.getEngineByMimeType("text/javascript");
getEngineByMimeType的参数可以是Mime types: [application/javascript, application/ecmascript, text/javascript,
text/ecmascript]中的任何一个,可以将text/javascript改成text/ecmascript。
- 根据名称得到脚本引擎
- ScriptEngine engine = manager.getEngineByName("javascript");
getEngineByName后的参数可以是Names: [js, rhino, JavaScript, javascript, ECMAScript, ecmascript]中的任何一个,
如可以将javascript改成ecmascript。
============================================================================================
上面已经讨论了执行脚本的第一步,就是得到一个可用的脚本引擎。在完成这项工作之后就可以利用这个脚本引擎执行相应的脚本了。我们可以使用ScriptEngine的eval方法来执行脚本。eval方法被重载的多次,但最常用的是 public Object eval(String script)。
下面的例子演示了如何使用eval方法来执行javascript脚本。
- import javax.script.*;
- import java.io.*;
- import static java.lang.System.*;
- public class FirstJavaScript
- {
- public static void main(String args[])
- {
- ScriptEngineManager manager = new ScriptEngineManager();
- // 得到javascript脚本引擎
- ScriptEngine engine = manager.getEngineByName("javascript");
- try
- {
- // 开始运行脚本,并返回当前的小时
- Double hour = (Double)engine.eval("var date = new Date();" +"date.getHours();");
- String msg;
- // 将小时转换为问候信息
- if (hour < 10)
- {
- msg = "上午好";
- }
- else if (hour < 16)
- {
- msg = "下午好";
- }
- else if (hour < 20)
- {
- msg = "晚上好";
- }
- else
- {
- msg = "晚安";
- }
- out.printf("小时%s: %s%n", hour, msg);
- }
- catch (ScriptException e)
- {
- err.println(e);
- }
- }
- }
上面的例子通过得到当前的小时,并将其转化为问候语。上面的程序的输出信息为:
- 小时9.0:上午好
这个例子最值得注意的是执行的2句脚本,最后一句是date.getHours()。并未将这个值赋给一个javascript变量。这时,eval方法就将这样的值返回。这有些类似C语言的(…)运算符。如(c=a+b, c + d),这个表达式的返回值是a+b+d。
=======================================================================================
和脚本语言进行交互
上面例子只是运行了一个非常简单的脚本。这个脚本是孤立的,并未通过Java向这脚本传递任何的值。虽然从这个脚本返回了一个值,但这种返回方式是隐式的。
脚本引擎除了这些简单的功能,还为我们提供了更强大的功能。甚至可以通过Java向脚本语言中传递参数,还可以将脚本语言中的变量的值取出来。这些功能要依靠ScriptEngine中的两个方法put和get。
put 有两个参数,一个是脚本变量名,另一个是变量的值,这个值是Object类型,因此,可以传递任何值。
get 有一个参数,就是脚本变量的名。
下面的代码通过javascript脚本将一个字符串翻转(这个字符串是通过java传给javascript的),然后通过java得到这个被翻转后的字符后,然后输出。
- import javax.script.*;
- import java.io.*;
- import static java.lang.System.*;
- public class ReverseString
- {
- public static void main(String args[])
- {
- ScriptEngineManager manager = new ScriptEngineManager();
- // 建立javascript脚本引擎
- ScriptEngine engine = manager.getEngineByName("javascript");
- try
- {
- // 将变量name和变量值abcdefg传给javascript脚本
- engine.put("name", "abcdefg");
- // 开始执行脚本
- engine.eval("var output ='' ;" +
- "for (i = 0; i <= name.length; i++) {" +
- " output = name.charAt(i) + output" +
- "}");
- // 得到output变量的值
- String name = (String)engine.get("output");
- out.printf("被翻转后的字符串:%s", name);
- }
- catch (ScriptException e)
- {
- err.println(e);
- }
- }
- }
以上代码的输出结果为:
- 被翻转后的字符串:gfedcba
==========================================================================================================================
让脚本运行得更快
众所周知,解释运行方式是最慢的运行方式。上述的几个例子无一例外地都是以解释方式运行的。由于Java EE 6的脚本引擎可以支持任何实现脚本引擎接口的语言。有很多这样的语言提供了编译功能,也就是说,在运行脚本之前要先将这些脚本进行编译(这里的编译一般将不是生成可执行文件,而只是在内存中编译成更容易运行的方式),然后再执行。如果某段脚本要运行之交多次的话,使用这种方式是非常快的。我们可以使用 ScriptEngine的compile方法进行编译。并不是所有脚本引擎都支持编译,只有实现了Compilable接口的脚本引擎才可以使用 compile进行编译,否则将抛出一个错误。下面的例子将演示如何使用compile方法编译并运行javascript脚本。
- import javax.script.*;
- import java.io.*;
- import static java.lang.System.*;
- public class CompileScript
- {
- public static void main(String args[])
- {
- ScriptEngineManager manager = new ScriptEngineManager();
- ScriptEngine engine = manager.getEngineByName("javascript");
- engine.put("counter", 0); // 向javascript传递一个参数
- // 判断这个脚本引擎是否支持编译功能
- if (engine instanceof Compilable)
- {
- Compilable compEngine = (Compilable)engine;
- try
- {
- // 进行编译
- CompiledScript script = compEngine.compile("function count() { " +
- " counter = counter +1; " +
- " return counter; " +
- "}; count();");
- out.printf("Counter: %s%n", script.eval());
- out.printf("Counter: %s%n", script.eval());
- out.printf("Counter: %s%n", script.eval());
- }
- catch (ScriptException e)
- {
- err.println(e);
- }
- }
- else
- {
- err.println("这个脚本引擎不支持编译!");
- }
- }
- }

上面的代码运行后的显示信息如下:
- Counter: 1.0
- Counter: 2.0
- Counter: 3.0
在这个例子中,先通过compile方法将脚本编译,然后通过eval方法多次进行调用。在这段代码中只有一个函数,因此,eval就返回了这个函数的值 。
=========================================================================================================================
动态调用脚本语言的方法
上面的例子只有一个函数,可以通过eval进行调用并将它的值返回。但如果脚本中有多个函数或想通过用户的输入来决定调用哪个函数,这就需要使用invoke方法进行动态调用。和编译一样,脚本引擎必须实现 Invocable接口 才可以动态调用脚本语言中的方法。下面的例子将演示如何通过动态调用的方式来运行上面的翻转字符串的javascript脚本。
- import javax.script.*;
- import java.io.*;
- import static java.lang.System.*;
- public class InvocableTest
- {
- public static void main(String args[])
- {
- ScriptEngineManager manager = new ScriptEngineManager();
- ScriptEngine engine = manager.getEngineByName("javascript");
- String name="abcdefg";
- if (engine instanceof Invocable)
- {
- try
- {
- engine.eval("function reverse(name) {" +
- " var output =' ';" +
- " for (i = 0; i <= name.length; i++) {" +
- " output = name.charAt(i) + output" +
- " } return output;}");
- Invocable invokeEngine = (Invocable)engine;
- Object o = invokeEngine.invokeFunction("reverse", name);
- out.printf("翻转后的字符串:%s", o);
- }
- catch (NoSuchMethodException e)
- {
- err.println(e);
- }
- catch (ScriptException e)
- {
- err.println(e);
- }
- }
- else
- {
- err.println("这个脚本引擎不支持动态调用");
- }
- }
======================================================================================================================
动态实现接口
脚本引擎还有一个更吸引的功能,那就是动态实现接口。如我们要想让脚本异步地执行,即通过多线程来执行,那InvokeEngine类必须实现 Runnable接口才可以通过Thread启动多线程。因此,可以通过getInterface方法来使InvokeEngine动态地实现 Runnable接口。这样一般可分为3步进行。
1. 使用javascript编写一个run函数
engine.eval("function run() {print(异步执行);}");
2. 通过getInterface方法实现Runnable接口
Runnable runner = invokeEngine.getInterface(Runnable.class);
3. 使用Thread类启动多线程
Thread t = new Thread(runner);
t.start();
下面是实现这个功能的详细代码。
- import javax.script.*;
- import static java.lang.System.*;
- public class InterfaceTest
- {
- public static void main(String args[])
- {
- ScriptEngineManager manager = new ScriptEngineManager();
- ScriptEngine engine = manager.getEngineByName("javascript");
- try
- {
- engine.eval("function run() {print(异步调用);}");
- Invocable invokeEngine = (Invocable)engine;
- Runnable runner = invokeEngine.getInterface(Runnable.class);
- Thread t = new Thread(runner);
- t.start();
- t.join();
- }
- catch (InterruptedException e)
- {
- err.println(e);
- }
- catch (ScriptException e)
- {
- System.err.println(e);
- }
- }
- }
java ScriptEngine 使用 (java运行脚本文件)的更多相关文章
- 【转】解决ubuntu13.10下,无法双击运行脚本文件
解决ubuntu13.10下,无法双击运行脚本文件 转自:http://www.aichengxu.com/other/975350.htm 首先,必须先设定好脚本的运行方法,当然如果只是she ...
- django项目中使用项目环境制作脚本 通过终端命令运行脚本文件
在实际的django项目开发中,有时候需要制作一些脚本文件对项目数据进行处理,然后通过终端命令运行脚本. 完整的实现流程如下: 1.在一个应用目录下(app, 必须是在应用目录下,可以专门创建一个应用 ...
- Scala学习笔记(二):运行脚本文件
在某个目录(如:F:\)下新建一个文本文件,命名为:hello.scala 其内容为: println("Hello World!") 那么这个时候该怎么运行这个脚本文件呢? 通过 ...
- 在Django中运行脚本文件以及打印出SQL语句。
Django终端打印SQL语句 在Django项目的settings.py文件中,在最后复制粘贴如下代码: LOGGING = { 'version': 1, 'disable_existing_lo ...
- expect 运行脚本文件 执行postgres数据库操作
#!/bin/bash /usr/bin/expect << EOF spawn /usr/local/pgsql/bin/.sh expect "*postgres:" ...
- 【Shell脚本】运行shell脚本文件的几种方法与区别
Shell脚本不同的运行方式会对当前Shell设置或者运行结果有所不同. 假设现在有一个脚本名为display_shell_script_args.sh,其内容如下: #!/home/pyf/bin/ ...
- Windows运行python脚本文件
开始学习python就是听说这个语言写脚本文件特别方便,简单使用.学了一段时间,但是直到现在我才直到直到怎么在Windows的cmd上运行脚本文件. 之前一直都是在pycharm上运行,并不实用. 百 ...
- Java基础笔记(1) 语言 JAVA的历史 Java的搭建环境
本文除了搭建是重点,其他的都当阅读小说一样去看就好了,不想看可以直接抓住重点,我会改变颜色勾出重点! 英语是人与人交流沟通的重要方式之一.JAVA:是人与计算机沟通交流重要方式之一.我们除了用java ...
- (八) .launch文件 ---编写简单的启动脚本文件
下面我们将介绍,如何编写一个启动脚本程序:(.launch文件) 还记得我们在 创建ROS软件包教程 中创建的第一个程序包(beginner_tutorials)吗,现在我们要使用它. 在 begin ...
随机推荐
- CDR是什么?CorelDRAW矢量绘图
CorelDRAW是矢量绘图软件 CorelDRAW Graphics Suite是加拿大Corel公司的平面设计软件: CorelDRAW 非凡的设计能力广泛地应用于商标设计.标志制作.模型绘制.插 ...
- JS 20180416作业
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- python海龟的使用
############################螺旋正方形############################### import turtle turtle.left(90)for i ...
- nmon分析文件各sheet含义
sheet名称sheet含义 SYS_SUMM系统汇总,蓝线为cpu占有率变化情况,粉线为磁盘IO的变化情况: AAA关于操作系统以及nmon本身的一些信息: BBBB系统外挂存储容量以及存储类型: ...
- 利用Selenium实现图片文件上传的两种方式介绍
在实现UI自动化测试过程中,有一类需求是实现图片上传,这种需求根据开发的实现方式,UI的实现方式也会不同. 一.直接利用Selenium实现 这种方式是最简单的一种实现方式,但是依赖于开发的实现. 当 ...
- HDU2147 kiki's game
/* HDU2147 kiki's game 博弈论 巴什博奕 http://acm.hdu.edu.cn/showproblem.php?pid=2147 题意:在一个n×m的棋盘上,初始棋子放在右 ...
- Java开源框架 iBase4J 搭建笔记
项目地址:https://gitee.com/iBase4J/iBase4J 搭建步骤: 1.git 代码 2.安装 Zookeeper 3.打包部署 Dubbo Admin ...
- pthread_cond 唤醒特定线程的方法
- POJ 2132
我早上调了一个早上,下午才发现把e=edge[e].next写成edge[e].next了... 这题直接DFS,一个剪枝是,当当前的最大质因数是最小公倍数的因数时,不用搜索 #include < ...
- HDU 4350
最近这些天,确实很烦恼.因为发现自己好像无论怎么样努力,也赶不上那些强校的学生.不得不承认,我们是传统弱校了.停了有一周了,什么也不想写,不停的反思,到底自己在哪里比不上人.D说,那是因为自始至终你只 ...