2016-06-09 142 views
4

我正在使用Java Jersey框架(使用Maven),并使用IntelliJ作为我的IDE。我遇到了这种运行时异常,只有当我尝试从命令行运行代码(使用maven编译,然后使用java -jar)时才会发生,但在IntelliJ中运行时却不是这样,这很奇怪。MessageBodyProviderNotFoundException从命令行运行jar时

我有一些Java代码,将努力使一个HTTP GET一些远程URL,并尝试读取返回的JSON到一些龙目POJO:

String targetUrl = "some valid URL"; 

WebTarget webTarget = client.target(targetUrl); 

Response response = webTarget.request(MediaType.APPLICATION_JSON_TYPE).get(); 

ParseResponse parseResponse = response.readEntity(ParseResponse.class); 

我不知道为什么,但是当它击中一个具有“readEntity()”方法,即最后一行,我会得到下面的错误:

org.glassfish.jersey.message.internal.MessageBodyProviderNotFoundException: MessageBodyReader not found for media type=text/json; charset=utf-8 

这很奇怪,因为我肯定在我的pom.xml指定的球衣,媒体JSON - 杰克逊依赖:

<dependency> 
     <groupId>org.glassfish.jersey.media</groupId> 
     <artifactId>jersey-media-json-jackson</artifactId> 
     <version>2.23</version> 
</dependency> 

而这正是我试图readEntity()到我的POJO类:

@Data 
@JsonIgnoreProperties(ignoreUnknown = true) 
public class ParseResponse { 

    @JsonProperty("id") 
    private Integer id; 

    ...Other params... 
} 

就像我之前提到的,奇怪的是,当我尝试在命令行中运行这只是发生像这样但在IntelliJ运行时没有错误:

mvn clean package 
java -jar target/NameOfJar.jar 

我在这里错过了一些明显的东西吗?我在网上查看过类似问题的其他人,但尚未找到解决方案。

感谢 IS

回答

9

如果你看看jersey-media-json-jackson罐子里面,你会看到一个文件

META-INF/services/org.glassfish.jersey.internal.spi.AutoDiscoverable 

这个文件的内容应该是一个类的单个全名实现的名称该文件,即

org.glassfish.jersey.jackson.internal.JacksonAutoDiscoverable 

该文件由Jersey auto-discoverable mechanism用于自动注册功能w没有我们必须明确注册它们。简而言之,它的工作原理是,所有具有应该自动注册组件的Jersey模块/ jar应该具有位于jar中的上述命名文件,其内容是可自动发现组件的名称。然后泽西将使用Service Loader pattern加载文件中指定的类并注册它们。

创建超级罐子时造成的问题是您只能拥有一个文件的副本,您不能有重复。那么如果我们有多个包含上述文件的罐子呢?那么这些文件中只有一个会被包含在uber jar中。哪一个?谁知道,但只有一个幸运的赢家。所以对于其余的罐子,他们的自动发现机制从来没有启动。这是杰克逊功能的情况下,自动发现的登录JacksonFeature。您可以尝试明确注册您的应用程序,并且您应该看到它现在可以正常工作。

但是可能有这个文件的其他罐子/模块呢?正因为如此,在创建超级罐子时,应该使用maven-shade-plugin。这个插件允许你做什么,是结合该文件的内容,使所有可发现性被包括到该单一文件。以下是一个示例用法

<plugin> 
    <groupId>org.apache.maven.plugins</groupId> 
    <artifactId>maven-shade-plugin</artifactId> 
    <version>2.3</version> 
    <configuration> 
     <createDependencyReducedPom>true</createDependencyReducedPom> 
     <filters> 
      <filter> 
       <artifact>*:*</artifact> 
       <excludes> 
        <exclude>META-INF/*.SF</exclude> 
        <exclude>META-INF/*.DSA</exclude> 
        <exclude>META-INF/*.RSA</exclude> 
       </excludes> 
      </filter> 
     </filters> 
    </configuration> 
    <executions> 
     <execution> 
      <phase>package</phase> 
      <goals> 
       <goal>shade</goal> 
      </goals> 
      <configuration> 
       <transformers> 
        <transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/> 
        <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer"> 
         <mainClass>com.example.YourApp</mainClass> 
        </transformer> 
       </transformers> 
      </configuration> 
     </execution> 
    </executions> 
</plugin> 

本示例实际上取自Dropwizard's Getting Started。你可以查看进一步的解释。关注的主要部分ServicesResorceTransformer,这是连接服务文件的内容。

+0

Jersey + Grizzly + Moxy,这个作品完美! – Alexandre