无法使用groovy注释中的类中定义的变量

Tom*_*Lin 5 groovy annotations gradle

我试图从Dropwizard示例中将一些代码从java移植到groovy.

我在java中看到,我可以使用以下代码而不会出现任何问题:

package com.example.helloworld;
import javax.ws.rs.*;
import javax.ws.rs.core.MediaType;

@Produces(MediaType.APPLICATION_JSON)
public class HelloWorldService{  

}
Run Code Online (Sandbox Code Playgroud)

但是,使用groovy编译器(1.8和2.0.6),类无法使用MediaType.APPLICATION_JSON周围的noClassFoundException进行编译

如果我更改此代码以使用实际的字符串值

@Produces('application/json')
public class HelloWorldService{  

}
Run Code Online (Sandbox Code Playgroud)

一切都很完美.

groovy解析注释的方式和java的方式有什么区别吗?

为了完整性,这是gradle项目的一部分,这是我的build.gradle(文件在src/groovy/com/example/helloworld下)

apply plugin: 'groovy'

// Set our project variables
project.ext {
    dropwizardVersion = '0.6.1'
}

repositories {
    mavenCentral()
}

dependencies {
    compile group: 'com.yammer.dropwizard', name: 'dropwizard-core', version: dropwizardVersion
    groovy group: 'org.codehaus.groovy', name: 'groovy-all', version: '1.8.7'
}
Run Code Online (Sandbox Code Playgroud)

编译错误是:

引起:java.lang.RuntimeException:java.lang.ClassNotFoundException:com.sun.ws.rs.ext.RuntimeDelegateImpl ... 17更多引起:java.lang.ClassNotFoundException:com.sun.ws.rs.ext. org.gradle.api.internal.tasks.compile.TransformingClassLoader.findClass(TransformingClassLoader.java:47)中的RuntimeDelegateImpl

Pet*_*ser 6

问题是由Groovy编译器的不幸限制引起的,即它使用反射来访问编译类路径上的类.这可能反过来触发其他类被加载,这可能在编译类路径上不可用.通常(但不总是)这些是运行时依赖性.

在具体的情况下,Groovy编译器javax.ws.rs.core.MediaType通过反射加载,最终导致com.sun.ws.rs.ext.RuntimeDelegateImpl通过Class.forName(由静态初始化器触发)加载,而不是在编译类路径上.解决方案是将该类放在编译类路径上.(从长远来看,解决方案是修复独立的Groovy编译器不使用反射,并且从我所知道的已经在队列中.)如果模块的传递依赖性不是问题,那么实现这一点的最简单方法是:

dependencies {
    compile "com.sun.jersey:jersey-client:1.15" 
}
Run Code Online (Sandbox Code Playgroud)

我怀疑Eclipse Groovy编译器没有这个问题,因为它不使用反射来访问编译类路径.我希望GMaven像Gradle一样爆炸,除非它被配置为使用Eclipse编译器(Gradle目前不支持).