从Java向Groovy对象添加属性

3
我希望您能够利用Groovy的元编程能力,将属性添加到Java中的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库方法来实现上述操作?

1个回答

6
因此,考虑到这个Groovy脚本:

...

def a = 'A String'

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

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

我们可以这样编写我们的Java装饰器类:
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脚本,则所有的断言都应该通过。


谢谢,那个完美地解决了!我不确定是怎么做到的,但 HandleMetaClass 似乎是将 Java 对象视为 Groovy 对象的关键。 - Alex Spurling

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接