Maven依赖项上的NoClassDefFoundError

时间:2012-05-12 23:45:42

标签: java maven maven-3

我第一次使用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>

任何人都可以帮助我吗?

7 个答案:

答案 0 :(得分:99)

默认情况下,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>1.6</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等,请参阅{{3}上的“示例”部分。 }。

答案 1 :(得分:20)

  

当我尝试运行它时,我得到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传统上为每个包装级别创建一个目录。

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

您的课程路径至少需要,并且没有换行符:

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上的分隔符是分号(;)。

我很抱歉没有及早发现它。问题出在你原来的帖子里,但我错过了。

答案 2 :(得分:3)

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

您可以使用插件包含相关的jar。

将以下代码段添加到您的pom的插件部分

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

答案 3 :(得分:2)

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

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 :(得分:0)

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

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

例如,A取决于B和C,而B取决于D:1.0,C取决于D:1.1,maven可能 只需导入D:1.0。如果A使用D:1.1而不是D:1.0的类,则会引发NoClassDefFoundError。

如果您也处于这种情况下,则需要解决依赖性冲突。

答案 5 :(得分:-2)

没有人提到,当有人弄乱POM文件中的版本号或删除对Maven服务器的依赖时,将发生此错误。没有书面解决方案,请更新此答案。

我可以通过使用-Dpackaging = class运行mvn install:install-file来解决此问题。然后按如下所述将条目添加到POM: https://stackoverflow.com/a/19066828/1308453

答案 6 :(得分:-4)

选择项目 - &gt;干净应该解决这个