java - 使用 ant 使用库创建可执行 jar 后出现 NoClassDefFoundError

标签 java jar ant build-script antbuilder

我创建了一个示例应用程序来演示我面临的问题。 我的示例应用程序包含带有 main 方法的单个类以及对 log4j 的依赖

package com.example.foo;

public class MainClass {
    private static final org.slf4j.Logger LOG = org.slf4j.LoggerFactory.getLogger(MainClass.class);

    public static void main(String[] args) {
        LOG.info("Hello Ant Build");
    }
}

我在 lib 文件夹中添加了两个 jar 文件 slf4j-api-1.7.25.jarslf4j-simple-1.7.25.jarProject Stricture

接下来,我创建了一个 ant 脚本来构建这个项目。 (这个项目在eclipse下运行成功,但这不是我需要的)

我的build.xml是

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<project name="sample-app" basedir="../SampleApp" default="jar">
    <property name="source.dir" value="src"/>
    <property name="lib.dir" value="lib"/>
    <property name="class.dir" value="bin"/>
    <property name="jar.dir" value="dist"/>
    <property name="jar.file" value="${jar.dir}/${ant.project.name}.jar"/>
    <property name="main-class" value="com.example.foo.MainClass"/>

    <path id="libraries.path">    
        <fileset dir="${lib.dir}">
            <include name="*.jar"/>
        </fileset>
    </path>

    <target name="clean" description="delete old files">
        <delete dir="${class.dir}"/>
        <delete dir="${jar.dir}"/>
    </target>

    <target name="compile" description="build class files" depends="clean">
        <mkdir dir="${class.dir}"/>
        <javac srcdir="${source.dir}" destdir="${class.dir}" includeantruntime="false">
            <classpath refid="libraries.path"/>
        </javac>
    </target>

    <target name="jar" depends="compile">
        <mkdir dir="${jar.dir}"/>
        <mkdir dir="${class.dir}/${lib.dir}"/>
        <copy todir="${class.dir}/${lib.dir}" flatten="true">
            <path refid="libraries.path"/>
        </copy>

        <manifestclasspath property="manifest.classpath" jarfile="${jar.file}">
            <classpath refid="libraries.path"/>
        </manifestclasspath>

        <jar destfile="${jar.file}" basedir="${class.dir}">
            <manifest>
                <attribute name="Main-Class" value="${main-class}"/>
                <attribute name="Class-Path" value="${manifest.classpath}"/>
            </manifest>
        </jar>  
    </target>
</project>

这样就成功创建了 jar 文件。所有必需的类文件和 jar 文件都存在于 jar 文件中。

Jar Contents

但是当我使用 java -jar example-app.jar 执行 jar 时,我得到 NoClassDefFoundError

$ java -jar sample-app.jar
java.lang.NoClassDefFoundError: org/slf4j/LoggerFactory
        at com.example.foo.MainClass.<clinit>(Unknown Source)
Caused by: java.lang.ClassNotFoundException: org.slf4j.LoggerFactory
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        ... 1 more
Exception in thread "main"

MANIFEST.MF 文件的内容是:

Manifest-Version: 1.0
Ant-Version: Apache Ant 1.10.1
Created-By: 1.8.0_111-b14 (Oracle Corporation)
Main-Class: com.example.foo.MainClass
Class-Path: ../lib/slf4j-api-1.7.25.jar ../lib/slf4j-simple-1.7.25.jar

我无法找出 jar 文件中的错误或缺失内容。请帮我解决我的 ant 脚本的问题。提前致谢。

最佳答案

标准类加载器不会从 jar 内部加载 jar 文件(请参阅 https://docs.oracle.com/javase/tutorial/deployment/jar/downman.html )。

使用 MANIFEST.MF 中给定的类路径条目,这些 jar 文件需要放置到文件夹“../lib”中,即执行可执行 jar 的目录的父文件夹中的“lib”文件夹。

其他解决方案是:

  1. 编写自定义类加载器
  2. 解压 slf4j jar 并重新打包内容

我不会推荐其中任何一个。

关于java - 使用 ant 使用库创建可执行 jar 后出现 NoClassDefFoundError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48007860/

相关文章:

java - 存储 selectOneListbox 中的上一个 SelectedValue

java - 无法执行 jar 文件 : "no main manifest attribute"

java - 构建机器 : error: package net. rim.device.api.ui 不存在

java - 如何在 ant build.xml 中为 javadoc 设置 -private

java - 获取图像时遇到问题(新手)

java - 定义一个数据输入检查方法并调用它没有帮助,android

java - 从 jar 运行 jar

java - 签署 jar 文件时,签名文件 (*.SF) 的用途是什么?

运行包含库 jar 的 jar 时出现 Java NoClassDefFoundError

java - 通过邮寄方式将文件上传到服务器 OutOfMemory