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/… 唯一的区别难道不是<groupId>org.apache.maven.plugins</groupId>
和<artifactId>maven-dependency-plugin</artifactId>
和<goals><goal>copy-dependencies</goal></goals>
。【参考方案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