java NoClassDefFoundError 与 Maven 依赖项 (log4j)

标签 java maven log4j noclassdeffounderror

我正在尝试使用 Maven 制作一个小程序。我正在使用大学教授提供的模板,并且我想使用 log4j 进行日志记录。 当我在 eclipse 中运行该程序时,它工作正常。但是,当我使用“mvn install”创建 jar 并尝试在 cmd 中运行该程序时,我收到 NoClassDefFoundError

C:\eclipse_workspace_buchhaltung\Buchhaltung\target>c:\jdk-11.0.4\bin\java.exe -jar Buchhaltung-1.0.0-SNAPSHOT.jar
Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/logging/log4j/LogManager
        at ch.hslu.demo.DemoKlasse.<clinit>(DemoKlasse.java:11)
Caused by: java.lang.ClassNotFoundException: org.apache.logging.log4j.LogManager
        at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:583)
        at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:178)
        at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:521)
        ... 1 more

我用来尝试的 DemoKlasse 如下所示:

package ch.hslu.demo;

import java.time.LocalDateTime;
import java.util.Scanner;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;

public class DemoKlasse {

    private static final Logger logger = LogManager.getLogger(DemoKlasse.class); // that's line 11

    public static void main(String[] args) {

        logger.info("DemoKlasse gestartet: " + LocalDateTime.now());

        System.out.println("Dies ist eine Testklasse!");
        String eingabe = "";
        Scanner sc = new Scanner(System.in);

        while(true) {

            System.out.println("Bitte geben Sie etwas ein. Beenden mit \"exit\"");

            eingabe = sc.nextLine();

            logger.info("Eingabe lautet: " + eingabe);

            if(eingabe.equalsIgnoreCase("exit")) {
                System.out.println("Das Programm wird beendet");
                logger.info("Exit wurde eingegeben: " + LocalDateTime.now());
                sc.close();
                System.exit(1);
            }

        }

    }

}

我的 pom 与 log4j 部分如下所示:

    <properties>
        ...
        <log4j2.version>2.12.1</log4j2.version>
        <junit5.version>5.5.1</junit5.version>
    </properties>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.apache.logging.log4j</groupId>
                <artifactId>log4j-api</artifactId>
                <version>${log4j2.version}</version>
                <scope>compile</scope>
            </dependency>
            <dependency>
                <groupId>org.apache.logging.log4j</groupId>
                <artifactId>log4j-core</artifactId>
                <version>${log4j2.version}</version>
                <scope>runtime</scope>
            </dependency>
            ...

我也在这里找到了这个:NoClassDefFoundError on Maven dependency 我尝试使用 maven-shade-plugin 和 maven- assembly-plugin 但仍然无法使其工作。

有人知道为什么我不能让这么简单的程序运行吗?

最佳答案

你在做什么?

您正在创建一个包含代码的 JAR 文件(以及指向您的主类的 Manifest 文件)。

问题是什么?

您正在将代码导出到 JAR 文件,但不会导出依赖项。显然,您的程序需要依赖项才能运行(日志库不是必需的,但其他库是)。

如何解决?

你说,你尝试了影子和程序集插件。您可以添加两者的依赖关系。我认为您只是配置错误。

使用maven-assemble-plugin,您需要创建一个jar-with-dependencies,并且您不应该忘记您的主类。

例如,您可以使用以下配置(将 complete-qualified-main-class 替换为主类的包名称,后跟主类的名称(用点分隔):

<plugin>
<artifactId>maven-assembly-plugin</artifactId>
    <configuration>
        <archive>
            <manifest>
                <mainClass>fully.qualified.MainClass</mainClass>
            </manifest> 
        </archive> 
        <descriptorRefs>
            <descriptorRef>jar-with-dependencies</descriptorRef>
        </descriptorRefs>
    </configuration>
    <executions>
        <execution>
            <id>make-assembly</id>
            <phase>package</phase>
            <goals>
                <goal>single</goal>
            </goals>
        </execution>
    </executions>
</plugin>

之后,您可以使用命令mvn package创建JAR。

确保执行 jar-with-dependencies 而不是由 maven-jar-plugin 创建的 JAR。

其他可能的问题

您的程序可能会使用 provided 范围声明 log4j。在这种情况下,log4j 不会包含在您的 jar-with-dependencies 中。

参见https://stackoverflow.com/a/574650/10871900

关于java NoClassDefFoundError 与 Maven 依赖项 (log4j),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59459084/

相关文章:

svn - 无法 mvn 发布 :prepare, 服务器证书验证失败

eclipse - 如何更改 eclipse 项目 .settings 目录下 org.eclipse.wst.common.project.facet.core.xml 上的 java 版本和 Web 版本

java - Log4j:如何配置多个附加程序:一个定向到控制台和一个文件,另一个严格定向到一个文件?

java - 我什么时候应该使用 error.printStrackTrace() 而不是 log4.j.error()

java - 当我在外部设备上运行应用程序时,为什么它给我的是 "unfortunatley <app name > has ended"

java - 我应该在我的 Java 应用程序中为 HTTP 使用 java.net 或 org.apache.http 库吗?

java - REST Jersey 客户端 - 无法将 JSON 解析为 POJO 类

java - log4j 的默认日志文件转储路径是什么

java - 如何从 Play Framework 的配置文件中读取对象列表

java - 无法订阅 Mono<XXX> spring webflux