3

我希望能够使用来自 Java 的 Groovy 元编程功能将属性添加到 String 对象的实例。

从 Groovy 执行此操作很简单:

class GroovyClass {
    def getDynamicString() {
        def myString = "hello"
        myString.metaClass.dynamicProperty = "there"
        return myString
    }
}

这是我的 Spock 测试用例:

def "should add dynamic property"() {
    GroovyClass groovyClass = new GroovyClass()

    when:
    def theString = groovyClass.getDynamicString()
    then:
    theString == "hello"
    theString.dynamicProperty == "there"
}

但是,我想从 Java 做同样的事情并通过相同的测试运行它。据我了解,Groovy 向 JDK 类(如 String)添加了额外的属性和方法(如 getMetaClass),但我不确定它是如何或何时这样做的。我发现以下解决方案有效,但初始化 GroovyShell 并将元编程代码编写为字符串似乎很麻烦。

public String getDynamicString()
{
    String myString = "hello";
    GroovyShell shell = new GroovyShell();
    shell.setVariable("myString", myString);
    shell.setVariable("theDynamicPropertyValue", "there");
    shell.evaluate("myString.metaClass.dynamicProperty = theDynamicPropertyValue");
    return myString;
}

有没有办法在不使用 shell.evaluate 的情况下完成上述操作 - 即直接从 Java 调用 Groovy 库方法?

4

1 回答 1

7

因此,鉴于这个 groovy 脚本:

def a = 'A String'

Decorator.decorate( a, 'foo', 'bar' )

assert a       == 'A String'
assert a.class == String
assert a.foo   == 'bar'

然后我们可以像这样编写我们的 java Decorator 类:

import org.codehaus.groovy.runtime.HandleMetaClass ;
import org.codehaus.groovy.runtime.InvokerHelper ;

public class Decorator {
  public static void decorate( Object o, String name, Object value ) {
    HandleMetaClass hmc = new HandleMetaClass( InvokerHelper.getMetaClass( o ), o ) ;
    hmc.setProperty( name, value ) ;
  }
}

然后如果我们编译 java 类并运行 groovy 脚本,所有的断言都应该通过

于 2013-03-18T13:36:40.657 回答