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);
  }
  }
  }
时间: 2024-08-07 16:37:36

java ScriptEngine 使用 (java运行脚本文件)的相关文章

【Java】windows下直接运行Java程序(利用bat运行jar文件)

这种方法只是掩耳盗铃,简化了运行jar的方式.并不是将Java代码打包成了exe文件. 具体步骤为,电脑上有JRE环境,然后利用bat文件包装一下运行jar的控制台语句,最后用C语言打包成exe文件. jar包有两种运行方式,一种为直接双击,直接双击需要配置注册表和环境变量才能运行jar包. 还有一种是在控制台运行,输入javaw -jar xxxx.所有的操作都需要Java环境. bat文件 bat是Windows下可以直接运行的批处理文件,编写也很简单. 新建一个txt文件,输入下面两行脚本

./(点斜杠)与. (点空格)运行脚本文件的差别

首先普及一个知识点:. (点空格)莫个脚本文件相当于利用source文件运行这个脚本. source命令使用方法: source FileName 作用:在当前bash环境下读取并运行FileName中的命令. 注:该命令通经常使用命令"."来替代. 如:source .bash_rc 与 . .bash_rc 是等效的. 注意:source命令与shell scripts的差别是, source在当前bash环境下运行命令,而scripts是启动一个子shell来运行命令. 这样假设

Scala学习笔记(二):运行脚本文件

在某个目录(如:F:\)下新建一个文本文件,命名为:hello.scala 其内容为: println("Hello World!") 那么这个时候该怎么运行这个脚本文件呢? 通过Windows的命令行窗口进入到Scala的安装目录下的bin目录下 输入“scala F:\hello.scala”,然后回车即可正确运行 请参考此文: scala在cmd模式下执行文件报<console>:1: error: ';' expected but '.' found. scalac

Windows运行python脚本文件

开始学习python就是听说这个语言写脚本文件特别方便,简单使用.学了一段时间,但是直到现在我才直到直到怎么在Windows的cmd上运行脚本文件. 之前一直都是在pycharm上运行,并不实用. 百度上说的已经很清楚了,但是还是需要自己亲手实验一下.http://jingyan.baidu.com/article/22fe7ced18776f3002617f2e.html 我刚开始也是加了环境变量了 啊,但是一直不好使,原来我的程序开头没有加上  #!usr/bin/python  表示该脚本

Java :无法将Java项识别为 cmdlet、函数、脚本文件或可运行程序的名称

使用命令行运行 Java 命令时出现错误:Java :无法将Java项识别为 cmdlet.函数.脚本文件或可运行程序的名称 如下所示: 这是因为:没有在系统变量path中配置JDK的安装路径 ,或者,你的电脑上还没有安装 Java 的JDK. JDK下载地址: Java SE Development Kit 14 Downloads 多种版本可供选择: 安装之后,需要 在系统变量path中配置JDK的安装路径. 把JDK的安装目录的路径 复制添加到系统变量的path, 如下: 现在就可以了:

用JSmooth制作java jar文件的可运行exe文件教程【图文】

这是我之前在个人博客3yj上面写的一篇文章,如今转载过来,原文地址 (这不是广告哦) 几年前,刚接触java的是,就想用一些方法把自己的劳动果实保护起来,曾经也用过非常多这种工具,有一个特别好用,今天写篇文章跟大家分享. 今天要用到的工具有 JSmooth 0.9.9-7 汉化版,能够到本站下载:JSmooth 0.9.9-7 汉化版 首先要把你的程序打包成双击能够运行的包,详细的方法我就不多说了,网上非常多的,主要命令是:jar cvfm test.jar *.* -C  之类的 主要是确保其

java程序执行SQL脚本文件

首先引入ibatis-common-2.jar包 import com.ibatis.common.jdbc.ScriptRunner; import com.ibatis.common.resources.Resources; jpetstore测试代码如下: package com.ibatis.jpetstore.test; import java.sql.DriverManager; import java.util.Properties; import com.ibatis.commo

【java】 linux下利用nohup后台运行jar文件包程序

Linux 运行jar包命令如下: 方式一: java -jar XXX.jar 特点:当前ssh窗口被锁定,可按CTRL + C打断程序运行,或直接关闭窗口,程序退出 那如何让窗口不锁定? 方式二 java -jar XXX.jar & &代表在后台运行. 特定:当前ssh窗口不被锁定,但是当窗口关闭时,程序中止运行. 继续改进,如何让窗口关闭时,程序仍然运行? 方式三 nohup java -jar XXX.jar & nohup 意思是不挂断运行命令,当账户退出或终端关闭时,

mac 下 使用 java运行 class 文件 总是提示 “错误: 找不到或无法加载主类”的解决方法

发现问题 切换到mac平台后,突然想写点程序运行在mac下,想到mac自带java,会方便好多.不过在这过程中遇到了麻烦: 总是提示 “错误: 找不到或无法加载主类” 工程结构 查了好久,终于找到原型所在,发现网上很多资料都写的都不太多,自己记录一下.先看看工程的位置和目录. 我的eclipse 工作空间位置是: /Users/zhangyunfei/workspace_java 我的工程名字叫:hellodemo 我的main class 的package name (包名)为: hellod