当前位置: 首页 > 面试题库 >

从命令行运行jar时出现MessageBodyProviderNotFoundException

贲凌
2023-03-14
问题内容

我正在使用Java
Jersey框架(与Maven),并使用IntelliJ作为我的IDE。我遇到了这个运行时异常,仅当我尝试从命令行运行代码(使用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依赖项:

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

这是我试图将Entity()读入的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


问题答案:

如果您查看jersey-media-json-jackson罐子内部,应该会看到一个文件

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

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

org.glassfish.jersey.jackson.internal.JacksonAutoDiscoverable

Jersey
自动发现机制使用此文件来自动注册功能,而无需我们明确注册它们。简而言之,它的工作原理是所有具有应自动注册的组件的Jersey模块/
jar都应在jar中包含上述命名的文件,其内容为可自动发现的组件的名称。然后,Jersey将使用Service
Loader模式加载文件中命名的类,并进行注册。

创建uber jar时,这导致的问题是,您只能拥有一个文件的副本,而不能有副本。那如果上面的文件有多个jar呢?那么这些文件中只有一个会被包含在uber
jar中。哪一个?谁知道,但只有一位幸运的获胜者。因此,对于其余的jars,它们的自动发现机制永远不会起作用。Jackson功能就是这种情况,在该功能中,自动发现功能会注册JacksonFeature。您可以尝试显式注册您的应用程序,并且应该看到它现在可以工作了。

但是其他可能有此文件的jar /模块呢?因此,在创建uber jar时,应使用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的《入门指南》。您可以查看它以获取进一步的解释。关注的主要部分是ServicesResorceTransformer,它是服务文件的连接。



 类似资料:
  • 我使用Java Jersey框架(与Maven一起使用),并使用IntelliJ作为IDE。我遇到过这个运行时异常,它只发生在尝试从命令行运行代码时(使用maven编译,然后使用Java-JAR),但在IntelliJ中运行时不会发生,这很奇怪。 我有一些Java代码,它将尝试使HTTP访问某个远程URL,并尝试将返回的JSON读入某个Lombok POJO: 正如我之前提到的,奇怪的是,只有当我

  • 问题内容: 好的,我知道这个问题以前已经被问过很多次了,但是我已经搜索了一下,查看了示例,并查看了过去一个月中关于SO的问题,我真的无法解决这个问题。我认为问题是我希望能够从Eclipse和命令行运行该程序。我也在使用OSX,我认为我正在阅读的许多示例都是针对Windows / Linux的。 如果我有一个要在命令行中运行的,在Eclipse中编译的简单程序,请执行以下操作: 我有另一个程序在Ec

  • 问题内容: 我在尝试使用sqlplus将sql脚本运行到oracle时遇到问题。该脚本仅填充一些虚拟数据: 使用Oracle SQL Developer运行脚本时,脚本运行良好,但是使用sqlplus命令行工具时,将输出以下内容,然后将其挂起: 我正在使用以下命令行运行该工具,该命令在其他脚本中也可以正常运行: 有任何想法吗?谢谢。 问题答案: 您需要在脚本的末尾放置一个,或以(例如重定向的输入代

  • 问题内容: 到目前为止,我主要使用eclipse。现在,我正在尝试从终端运行Java,但程序包存在问题。 这是我的文件: 我使用编译此代码,然后运行,它给我: 当我删除一切正常。我想念什么? 给出: 问题答案: 您需要在一个目录级别上运行java命令,并以完全合格的软件包名称提供它,例如: 请参阅Java Launcher如何查找用户类 以了解其工作方式。

  • 问题内容: 当前,在用于创建用于分发代码的jar的工作流中,我: 在Eclipse中右键单击我的项目 选择“导出” 选择“ JAR文件” 取消选中顶级文件,例如.classpath,.project 仅检查“导出生成的类文件” 点击“完成” 这样可以确保.class文件是最新的并创建jar。我想从命令行执行相同的操作,但是大多数用于创建jar的文档似乎只是在破坏已经存在的文件,而没有创建.clas

  • 我遇到了从命令提示符运行java程序的问题。我有一个名为DataRecover的java文件,还有一个名为Triple的java文件。现在,当我在命令提示符下运行javac triple.java时,它会执行它应该执行的操作。但是,当我运行javac DataRecover.java时,它会出现这样的错误消息:“线程”main“java.lang.noClassDeffounderRror:Dat