从命令行运行jar时出现MessageBodyProviderNotFoundException

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

我有一些Java代码将尝试在某个远程URL上进行HTTP GET并尝试将返回的JSON读入某些Lombok 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中指定了jersey-media-json-jackson依赖项:

  org.glassfish.jersey.media jersey-media-json-jackson 2.23  

这是我尝试将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 

我错过了一些明显的东西吗? 我曾在网上看过其他有类似问题的人,但还没有找到解决方案。

谢谢你

如果你看看jersey-media-json-jackson jar里面你应该看到一个文件

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

此文件的内容应该是实现文件名的类的单个完全限定名称,即

 org.glassfish.jersey.jackson.internal.JacksonAutoDiscoverable 

Jersey 自动发现机制使用此文件自动注册function,而无需我们显式注册它们。 简而言之,它的工作原理是,所有具有应该自动注册的组件的Jersey模块/ jar应该具有位于jar中的上述命名文件,其内容是可自动发现的组件的名称。 然后,Jersey将使用Service Loader模式加载文件中指定的类,并注册它们。

创建超级jar时导致的问题是,您只能拥有一个文件副本,不能重复。 那么如果我们有多个上述文件的jar子怎么办? 那么只有其中一个文件将包含在超级jar中。 哪一个? 谁知道,但只有一个幸运的赢家。 所以对于其他的jar子来说,他们的自动发现机制永远不会开始。这就是你的jackson特征的情况,其中可自动发现的东西注册了JacksonFeature 。 您可以尝试明确注册您的应用程序,您应该看到它现在有效。

但是可能有这个文件的其他jar子/模块呢? 出于这个原因,在创建超级jar时,你应该使用maven-shade-plugin 。 此插件允许您执行的操作是组合文件的内容,以便将所有可发现的内容包含在该单个文件中。 以下是一个示例用法

  org.apache.maven.plugins maven-shade-plugin 2.3  true   *:*  META-INF/*.SF META-INF/*.DSA META-INF/*.RSA       package  shade      com.example.YourApp       

这个例子实际上来自Dropwizard的入门 。 您可以查看它以获得进一步的解释。 关注ServicesResorceTransformer的主要部分是连接服务文件的内容。