【转】Java中用groovy

http://tglnn.iteye.com/blog/376978

 

很多情况下,Groovy是执行某一类任务的理想工具,如快速原型开发设计(rapid prototyping)或创建可由宏指令(macros)或插件(plug-ins)扩展的模块应用。这些扩展可以用Groovy实现,而不需冗长乏味的开发部署周期便可嵌入到您的应用程序中。相信,Groovy极富表现力,其简洁性和强大特性会给您的程序开发带来很大的好处。

在其他情况下,Groovy或许不是最好的选择。这一点对高性能要求极高的应用尤其适用,因为在灵活性和运行速度之间权衡取舍是不可避免的。

Groovy最大的亮点便是与Java的完美集成。Groovy的灵活性和机动性,让开发者至少可以通过5种方式实现与Java集成,当然各种方式均有其优缺点。下面章节将详细讨论这5种集成方式,并给出各种方式的适用条件。

编译成字节码(Compiling to Bytecode)

最简单直接的集成方法便是把Groovy文件编程成Java字节码(即.class文件),并能够在Java类路径(class path)获取这些文件。这种方式也有弊端:一方面您必须首先编译所有的Groovy文件,而与此同时,部分Groovy文件中引用了其他的Java类同样也需要先被编译,这时便会出现编译问题。

使用GroovyShell(Using GroovyShell)

GroovyShell允许在Java类中(甚至Groovy类)求任意Groovy表达式的值。您可使用Binding对象输入参数给表达式,并最终通过GroovyShell返回Groovy表达式的计算结果。清单2.19展示了如何使用GroovyShell。

清单2.19 GroovyShell应用

 

import groovy.lang.Binding;

import groovy.lang.GroovyShell;

 

public class GroovyShellExample {

 

  public static void main(String args[]) {

    Binding binding = new Binding();

    binding.setVariable("x", 10);

    binding.setVariable("language", "Groovy");

    GroovyShell shell = new GroovyShell(binding);

    Object value = shell.evaluate

      ("println \"Welcome to $language\"; y = x * 2; z = x * 3; return x ");

    assert value.equals(10);

    assert binding.getVariable("y").equals(20);

    assert binding.getVariable("z").equals(30);

  }

}

 

GroovyShell是推求动态类型表达式的有效工具。典型的应用就是推求用户输入Groovy动态表达式的值,通常会有一个用户界面(user interface,UI),如电子表格程序(spreadsheet application)。

使用GroovyScriptEngine(Using GroovyScriptEngine)

GroovyShell多用于推求对立的脚本或表达式,如果换成相互关联的多个脚本,使用GroovyScriptEngine会更好些。GroovyScriptEngine从您指定的位置(文件系统,URL,数据库,等等)加载Groovy脚本,并且随着脚本变化而重新加载它们。如同GroovyShell一样,GroovyScriptEngine也允许您传入参数值,并能返回脚本的值。

假设您有一个简单的Groovy 脚本,C:\tmp\SimpleScript.groovy,具体如下:

//SimpleScript.groovy

println "Welcome to $language"

return "The End"

 

清单2.20演示了如何传入参数给GroovyScriptEngine,并执行SimpleScript.groovy脚本,然后返回一个值。

 

清单2.20 GroovyScriptEngineExample.java

import groovy.lang.Binding;

import groovy.util.GroovyScriptEngine;

 

public class GroovyScriptEngineExample {

 

  public static void main(String args[]) {

    try {

      GroovyScriptEngine engine = new GroovyScriptEngine("");

      Binding binding = new Binding();

      binding.setVariable("language", "Groovy");

      Object value = engine.run("SimpleScript.groovy", binding);

      assert value.equals("The End");

    } catch (Exception e) {

      e.printStackTrace();

    }

  }

}

 

尽管GroovyScriptEngine是执行多个Groovy脚本的有效方法,但仍不能妥善处理复杂类。对于同时处理Groovy类和脚本的情况,最完善的解决方案便是GroovyClassLoader(其实,GroovyShell和GroovyScriptEngine都会用到它)

 

使用GroovyClassLoader(Using GroovyClassLoader)

GroovyClassLoader是一个定制的类装载器,负责解释加载Java类中用到的Groovy类。它也能编译。清单2.21展示了如何使用GroovyClassLoader加载Groovy类并且调用该类的一个方法。

 

清单2.21 GroovyClassLoader应用

//GroovySimpleFileCreator.groovy

 

class GroovySimpleFileCreator {

  public createFile(String fileName){

    File file = new File(fileName);

    file.createNewFile();

  }

}

 

//GroovyClassLoaderExample.java

 

import groovy.lang.GroovyClassLoader;

import groovy.lang.GroovyObject;

 

import java.io.File;

 

public class GroovyClassLoaderExample{

 

  public static void main(String args[]) {

    try {

      GroovyClassLoader loader = new GroovyClassLoader();

      Class fileCreator = loader.parseClass

        (new File("GroovySimpleFileCreator.groovy"));

      GroovyObject object = (GroovyObject) fileCreator.newInstance();

      object.invokeMethod("createFile", "C:\\temp\\emptyFile.txt");

    } catch (Exception e) {

      e.printStackTrace();

    }

  }

}

 

使用GroovyClassLoader典型的情景之一便是:您有一个Java接口和一个实现该Java接口的Groovy类。因此,您可以使用GroovyClassLoader加载Groovy实现类到应用中,这样便可直接调用接口的方法了,清单2.22给予形象地说明。

清单2.22 在Groovy中实现Java接口

//Shape.java:

 

public interface Shape{

  int calculateArea();

}

 

//Square.groovy:

 

class Square implements Shape {

 

  def x;

  int calculateArea(){

    return x * x;

  }

}

 

// GroovyClassLoaderExample2.java

 

import groovy.lang.GroovyClassLoader;

import groovy.lang.GroovyObject;

 

import java.io.File;

 

public class GroovyClassLoaderExample2{

 

  public static void main(String args[]) {

    try {

      GroovyClassLoader loader = new GroovyClassLoader();

      Class groovyClass = loader.parseClass(new File("Square.groovy"));

      GroovyObject object = (GroovyObject) groovyClass.newInstance();

      object.invokeMethod("setX", 10);

      Shape shape = (Shape) object;

      assert shape.calculateArea() == 100;

    } catch (Exception e) {

      e.printStackTrace();

    }

  }

}

 

使用JSR 223(Using JSR 223)

如果您正在使用Java 6,就可以选择使用Sun’s Java Specification Request(JSR) 223:Scripting for the Java Platform。使用JSR 223可使您的应用与特定的脚本引擎实现很好的分离,使您轻松更换脚本语言。如果您想在Java代码中使用其他的脚本语言(如BeanShell或JRuby),还是推荐使用JSR 223。如果您没有使用Java 6而且还想选择多脚本语言混合编程,可以参考Apache’s Bean Scripting框架:http://jakarta.apache.org/bsf。除非要做到您的应用与特定的脚本引擎很好的解耦,否则使用Groovy自带的方式相对比较灵活些。

清单2.23展示了如何使用JSR223集成Groovy。您必须保证groovy-engine.jar文件已经在您的类路径中。您可到https://scripting.dev.java.net下载该jar文件。这个例子演示了如何调用Groovy方法,传递参数和返回一个值。

 

清单2.23 使用JSR 223

import javax.script.Invocable;

import javax.script.ScriptEngine;

import javax.script.ScriptEngineManager;

 

public class GroovyJSR223Example {

 

  public static void main(String args[]) {

    try {

      ScriptEngineManager factory = new ScriptEngineManager();

      ScriptEngine engine = factory.getEngineByName("groovy");

      String HelloLanguage = "def hello(language) {return \"Hello $language\"}";

      engine.eval(HelloLanguage);

      Invocable inv = (Invocable) engine;

      Object[] params = { new String("Groovy") };

      Object result = inv.invokeFunction("hello", params);

      assert result.equals("Hello Groovy");

    } catch (Exception e) {

      // TODO Auto-generated catch block

      e.printStackTrace();

    }

  }

}

猜你喜欢

转载自gcc2ge.iteye.com/blog/2198670