1

我对 Groovy 完全陌生,所以我希望答案不明显......

假设我有一个脚本“Test.groovy”:

class A {
    def greet() {println "Hey there!"}
}

new A().greet()

我用GroovyShell(来自Java)评估这个脚本:

new GroovyShell().evaluate(new File("Test.groovy"));

我得到了预期的输出:

嘿!

现在,我从脚本中删除最后一行,而是在单独的调用中评估它evaluate(),我得到一个非常模糊的异常。

“测试.groovy”:

class A {
    def greet() {println "Hey there!"}
}

爪哇:

GroovyShell shell = new GroovyShell();
shell.evaluate(new File("Test.groovy"));
shell.evaluate("new A().greet()");

org.codehaus.groovy.runtime.metaclass.MissingMethodExceptionNoStack:没有方法签名:A.main() 适用于参数类型:([Ljava.lang.String;) 值:[[]] 可能的解决方案:wait(),等待(long),any(),find(),等待(long,int),每个(groovy.lang.Closure)

更有趣的是,如果我让脚本保持原样并只更改 Java 部分,它会完美运行(我得到两个“嘿!”)

4

1 回答 1

5

这应该有助于解释您所看到的:http: //www.groovy-lang.org/structure.html#_script_class

Groovy 将您的第一个 .groovy 文件视为脚本,因为最后一行存在于类声明之外。Groovy 编译成 Java 字节码,而 Java 要求所有代码都定义在一个类中。为了遵守,Groovy 做了一些魔术,并使用方法将您的脚本动态转换为 Java 类main——类似于以下内容:

public class script1440427072752 extends groovy.lang.Script { 

    public script1440427072752() {
    }

    public script1440427072752(groovy.lang.Binding context) {
        super(context)
    }

    public static void main(java.lang.String[] args) {
        org.codehaus.groovy.runtime.InvokerHelper.runScript(script1440427072752, args)
    }

    public java.lang.Object run() {
        new A().greet()
    }

}
public class A extends java.lang.Object { 

    public java.lang.Object greet() {
        this.println('Hey there!')
    }

}

然而,当您删除该行时,Groovy 将您的 .groovy 文件视为一个典型的 Java 类,名为A. 无需动态翻译groovy.lang.Script

当您尝试执行A时,GroovyShell 寻找一种main方法,但找不到,并抛出该错误。

于 2015-08-24T14:43:19.170 回答