Maven 依赖的 NoClassDefFoundError

Posted

技术标签:

【中文标题】Maven 依赖的 NoClassDefFoundError【英文标题】:NoClassDefFoundError on Maven dependency 【发布时间】:2012-05-21 01:10:10 【问题描述】:

我第一次使用 Maven,但我被依赖项困住了。

我使用 Eclipse 创建了一个 Maven 项目并添加了依赖项,它运行正常。

但是当我尝试通过命令行运行它时:

$ mvn package  # successfully completes
$ java -cp target/bil138_4-0.0.1-SNAPSHOT.jar tr.edu.hacettepe.cs.b21127113.bil138_4.App # NoClassDefFoundError for dependencies

它下载依赖项,成功构建,但是当我尝试运行它时,我得到 NoClassDefFoundError:

Exception in thread "main" java.lang.NoClassDefFoundError: org/codehaus/jackson/JsonParseException
        at tr.edu.hacettepe.cs.b21127113.bil138_4.db.DatabaseManager.<init>(DatabaseManager.java:16)
        at tr.edu.hacettepe.cs.b21127113.bil138_4.db.DatabaseManager.<init>(DatabaseManager.java:22)
        at tr.edu.hacettepe.cs.b21127113.bil138_4.App.main(App.java:10)
Caused by: java.lang.ClassNotFoundException: org.codehaus.jackson.JsonParseException
        at java.net.URLClassLoader$1.run(URLClassLoader.java:217)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:205)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:321)
        at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:294)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:266)
        ... 3 more

我的 pom.xml 是这样的:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>

  <groupId>tr.edu.hacettepe.cs.b21127113</groupId>
  <artifactId>bil138_4</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  <packaging>jar</packaging>

  <name>bil138_4</name>
  <url>http://maven.apache.org</url>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  </properties>

  <dependencies>        
    <dependency>
            <groupId>org.codehaus.jackson</groupId>
            <artifactId>jackson-core-asl</artifactId>           
    </dependency>
        <dependency>
            <groupId>org.codehaus.jackson</groupId>
            <artifactId>jackson-mapper-asl</artifactId>             
        </dependency>
  </dependencies>
  <dependencyManagement>
    <dependencies>
    <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-core-asl</artifactId>
        <version>1.9.6</version>
    </dependency>
    <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-mapper-asl</artifactId>
        <version>1.9.6</version>
    </dependency>
    </dependencies>
  </dependencyManagement>
</project>

谁能帮帮我?

【问题讨论】:

我遇到了同样的问题,我忘记了主类中的包 com.myapp.mypack.myclass。添加它解决了我的问题。 【参考方案1】:

默认情况下,Maven 不会在它构建的 JAR 文件中捆绑依赖项,并且当您尝试在命令行执行 JAR 文件时,您不会在类路径中提供它们。这就是为什么 Java VM 在尝试执行代码时找不到库类文件的原因。

您可以使用-cp 参数手动指定类路径上的库,但这很快就会变得很烦人。

更好的解决方案是将库代码“隐藏”到您的输出 JAR 文件中。有一个名为 maven-shade-plugin 的 Maven 插件可以执行此操作。您需要在 POM 中注册它,当您运行 mvn package 时,它会自动构建一个“uber-JAR”,其中包含您的类和库代码的类。

要简单地捆绑所有必需的库,请将以下内容添加到您的 POM:

<project>
  ...
  <build>
    <plugins>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-shade-plugin</artifactId>
        <version>3.2.4</version>
        <executions>
          <execution>
            <phase>package</phase>
            <goals>
              <goal>shade</goal>
            </goals>
          </execution>
        </executions>
      </plugin>
    </plugins>
  </build>
  ...
</project>

完成后,您可以重新运行上面使用的命令:

$ mvn package
$ java -cp target/bil138_4-0.0.1-SNAPSHOT.jar tr.edu.hacettepe.cs.b21127113.bil138_4.App

如果您想根据应包含的 JAR 对 shade 插件进行进一步配置,为可执行 JAR 文件指定 Main-Class,等等,请参阅maven-shade-plugin site 上的“示例”部分。

【讨论】:

如果有人遇到这个答案,您可以在这里获得插件的实际版本,因为 1.6 现在已经很老了:mvnrepository.com/artifact/org.apache.maven.plugins/… 唯一的区别难道不是 3.1.1(而不是 1.6)吗?运行“mvn package”似乎会自动下载 maven-shade 我的团队碰巧使用&lt;groupId&gt;org.apache.maven.plugins&lt;/groupId&gt;&lt;artifactId&gt;maven-dependency-plugin&lt;/artifactId&gt;&lt;goals&gt;&lt;goal&gt;copy-dependencies&lt;/goal&gt;&lt;/goals&gt;【参考方案2】:

当我尝试运行它时,我得到 NoClassDefFoundError

如何运行?您可能试图在没有正确导入 Maven 类路径的情况下使用 eclipse 运行它。请参阅 m2eclipse 插件以将 maven 与 eclipse 集成。

要验证您的 Maven 配置是否正确,您可以使用 exec plugin 运行您的应用程序:

mvn exec:java -D exec.mainClass=<your main class>

更新:首先,关于你在运行exec:java 时的错误,你的主类是tr.edu.hacettepe.cs.b21127113.bil138_4.App。在谈论类名时,它们(几乎)总是用点分隔。简单的类名只是最后一部分:App 在您的情况下。完全限定的名称是完整的包加上简单的类名,当你想运行某些东西时,这就是你给 maven 或 java 的名称。您尝试使用的是源文件的文件系统路径。那是完全不同的野兽。与文件系统中的源文件相比,类名通常直接转换为在类路径中找到的类文件。在您的特定情况下,有问题的类文件可能位于 target/classes/tr/edu/hacettepe/cs/b21127113/bil138_4/App.class,因为 maven 编译为 target/classes,而 java 传统上为每个级别的打包创建一个目录。

您最初的问题只是您没有将杰克逊罐子放在您的课​​程路径上。当你从命令行运行一个 java 程序时,你必须设置类路径,让它知道它可以从哪里加载类。您已经添加了自己的 jar,但没有添加其他必需的 jar。您的评论让我觉得您不了解如何手动构建类路径。简而言之,类路径可以有两件事:包含类文件的目录和包含类文件的 jar。包含 jar 的目录将不起作用。有关构建类路径的更多详细信息,请参阅“Setting the class path”以及java 和javac 工具文档。

你的类路径至少需要是,并且没有换行符:

target/bil138_4-0.0.1-SNAPSHOT.jar:
/home/utdemir/.m2/repository/org/codehaus/jackson/jackson-core-asl/1.9.6/jackson-core-asl-1.9.6.jar:
/home/utdemir/.m2/repository/org/codehaus/jackson/jackson-mapper-asl/1.9.6/jackson-mapper-asl-1.9.6.jar

请注意,Windows 上的分隔符是分号 (;)。

我很抱歉没有早点注意到它。问题出在您原来的帖子中,但我错过了。

【讨论】:

收到此错误:pastebin.com/g2txkMY6。我可以使用 m2eclipse 插件在 eclipse 上运行它,我在上面开发了应用程序。我无法弄清楚如何通过命令行运行它。以及如何导入我的 Maven 类路径?编辑:尝试将 -classpath /home/utdemir/.m2/repository/ 参数添加到 java 命令,仍然没有运气。 谢谢。如您所说设置类路径解决了问题,将问题标记为已解决。但我还有几个问题: 1- 为什么我不必将 Jackson 添加到 exec 插件的类路径中? 2- Exec 插件需要编译我的项目,所以我应该运行 mvn compile 或 package,对吗? 3- 这段代码是我的任务,在他们的类路径中添加库是他们的责任,还是我必须做点什么? 1) 几乎所有通过 maven 运行的东西,包括 exec 插件,都会根据 pom.xml 中指定的项目依赖关系自动为您设置类路径。这是maven受欢迎的一个主要原因。 2) 是的,在运行 exec 插件之前,您必须至少运行 mvn compile。 3) 最后,如果我要提交作业,我会确保它可以以任何指定的方式完全执行。如果您对项目应如何打包或上交有疑问,我建议您开始一个新的 SO 问题。 插件链接失效。【参考方案3】:

您必须在 pom 文件中为您的依赖项创建类路径。因此,您必须将所有依赖项复制到一个位置。

Check my blog.

<build>
  <plugins>
    <plugin>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-dependency-plugin</artifactId>
      <version>2.1</version>
      <executions>
        <execution>
          <id>copy-dependencies</id>
          <phase>package</phase>
          <goals>
            <goal>copy-dependencies</goal>
          </goals>
          <configuration>
            <outputDirectory>$project.build.directory/lib</outputDirectory>
            <overWriteReleases>false</overWriteReleases>
            <overWriteSnapshots>false</overWriteSnapshots>
            <overWriteIfNewer>true</overWriteIfNewer>
          </configuration>
        </execution>
      </executions>
    </plugin>
    <plugin>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-jar-plugin</artifactId>
      <version>2.4</version>
      <configuration>
        <archive>
          <manifest>
            <addClasspath>true</addClasspath>
            <classpathPrefix>lib/</classpathPrefix>
            <mainClass>$fullqualified path to your main Class</mainClass>
          </manifest>
        </archive>
      </configuration>
    </plugin>
  </plugins>
</build>

【讨论】:

【参考方案4】:

这是由于 Morphia jar 不是您的输出 war/jar 的一部分。 Eclipse 或本地构建将它们作为类路径的一部分包含在内,但远程构建或自动/计划构建不将它们视为类路径的一部分。

您可以使用插件包含依赖的 jar。

将下面的 sn-p 添加到你的 pom 的插件部分

    <plugin>
        <artifactId>maven-assembly-plugin</artifactId>
        <version>3.0.0</version>
        <configuration>
            <descriptorRefs>
                <descriptorRef>jar-with-dependencies</descriptorRef>
            </descriptorRefs>
        </configuration>
    </plugin>

【讨论】:

【参考方案5】:

由于某种原因,该库在编译时存在,但在运行时丢失

我的情况是,一个库的两个版本冲突。

例如A依赖B和C,而B依赖D:1.0,C依赖D:1.1,maven可以 只需导入 D:1.0。如果 A 使用了一个在 D:1.1 中但不在 D:1.0 中的类,则会抛出 NoClassDefFoundError。

如果你也遇到这种情况,需要解决依赖冲突。

【讨论】:

为了帮助提问者的这个答案,如果您解释了如何解决所提到的冲突类型,它可能会有所帮助。【参考方案6】:

我可以通过运行 mvn install:install-file 和 -Dpackaging=class 来解决这个问题。然后按照here 的描述向 POM 添加条目:

【讨论】:

【参考方案7】:

选择项目 -> Clean 应该可以解决这个问题

【讨论】:

以上是关于Maven 依赖的 NoClassDefFoundError的主要内容,如果未能解决你的问题,请参考以下文章

当我尝试保存测试计划时,为啥会收到 NoClassDefFound 错误?

Android 应用程序在 SDK-tools 更新版本后崩溃(NoClassDefFound,工具版本 22)

XML 代码运行正常,但 junit 因 NoClassDefFound 而失败

错误:发生 JNI 错误,请检查您的安装并重试。NoClassDefFound 错误:DRPCExecutionException

在 android 2.3 上使用 jcifs 库时出现 NoClassDefFound 异常

XStream noClassDefFound 错误取决于运行时