import java.io.*;
import javax.script.*;
public class App
{
public static void main(String[] args)
{
try
{
ScriptEngine engine =
new ScriptEngineManager().getEngineByName("javascript");
for (String arg : args)
{
FileReader fr = new FileReader(arg);
engine.eval(fr);
}
}
catch(IOException ioEx)
{
ioEx.printStackTrace();
}
catch(ScriptException scrEx)
{
scrEx.printStackTrace();
}
}
}
eval()方法也可以直接操作一个String,所以这个脚本不一定必须是文件系统的一个文件—它可以来自于数据库、用户输入,或者甚至可以基于环境和用户操作在应用程序中生成。
4.将Java对象绑定到脚本空间
仅仅调用一个脚本还不够:脚本通常会与Java环境中创建的对象进行交互。这时,Java主机环境必须创建一些对象并将它们绑定,这样脚本就可以很容易找到和使用这些对象。这个过程是ScriptContext对象的任务,如清单6所示:
清单6.为脚本绑定对象
import java.io.*;
import javax.script.*;
public class App
{
public static void main(String[] args)
{
try
{
ScriptEngine engine =
new ScriptEngineManager().getEngineByName("javascript");
for (String arg : args)
{
Bindings bindings = new SimpleBindings();
bindings.put("author", new Person("Ted", "Neward", 39));
bindings.put("title", "5 Things You Didn't Know");
FileReader fr = new FileReader(arg);
engine.eval(fr, bindings);
}
}
catch(IOException ioEx)
{
ioEx.printStackTrace();
}
catch(ScriptException scrEx)
{
scrEx.printStackTrace();
}
}
}
访问所绑定的对象很简单—所绑定对象的名称是作为全局命名空间引入到脚本的,所以在Rhino中使用Person很简单,如清单7所示:
清单7.
println("Hello from inside scripting!")
println("author.firstName = " + author.firstName)
您可以看到,JavaBeans样式的属性被简化为使用名称直接访问,这就好像它们是字段一样。
5.编译频繁使用的脚本
脚本语言的缺点一直存在于性能方面。其中的原因是,大多数情况下脚本语言是“即时”解译的,因而它在执行时会损失一些解析和验证文本的时间和CPU周期。运行在JVM的许多脚本语言最终会将接收的代码转换为Java字节码,至少在脚本被第一次解析和验证时进行转换;在Java程序关闭时,这些即时编译的代码会消失。将频繁使用的脚本保持为字节码形式可以帮助提升可观的性能。
我们可以以一种很自然和有意义的方法使用Java Scripting API。如果返回的ScriptEngine实现了Compilable接口,那么这个接口所编译的方法可用于将脚本(以一个String或一个Reader传递过来的)编译为一个CompiledScript实例,然后它可用于在eval()方法中使用不同的绑定重复地处理编译后的代码,如清单8所示:
清单8.编译解译后的代码
import java.io.*;
import javax.script.*;
public class App
{
public static void main(String[] args)
{
try
{
ScriptEngine engine =
new ScriptEngineManager().getEngineByName("javascript");
for (String arg : args)
{
Bindings bindings = new SimpleBindings();
bindings.put("author", new Person("Ted", "Neward", 39));
bindings.put("title", "5 Things You Didn't Know");
FileReader fr = new FileReader(arg);
if (engine instanceof Compilable)
{
System.out.println("Compiling....");
Compilable compEngine = (Compilable)engine;
CompiledScript cs = compEngine.compile(fr);
cs.eval(bindings);
}
else
engine.eval(fr, bindings);
}
}
catch(IOException ioEx)
{
ioEx.printStackTrace();
}
catch(ScriptException scrEx)
{
scrEx.printStackTrace();
}
}
}
在大多数情况中,CompiledScript实例需要存储在一个长时间存储中(例如,servlet-context),这样才能避免一次次地重复编译相同的脚本。然而,如果脚本发生变化,您就需要创建一个新的CompiledScript来反映这个变化;一旦编译完成,CompiledScript就不再执行原始的脚本文件内容。
结束语
Java Scripting API在扩展Java程序的范围和功能方面前进了很大一步,并且它将脚本语言的编码效率的优势带到Java环境。jrunscript—它显然不是很难编写的程序—以及javax.script给Java
本文来源:ITeye博客 作者:佚名