eclipse 在用 Ant 编译的文件中找不到主类

声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow 原文地址: http://stackoverflow.com/questions/3143567/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me): StackOverFlow

提示:将鼠标放在中文语句上可以显示对应的英文。显示中英文
时间:2020-09-19 14:51:14  来源:igfitidea点击:

Cannot find Main Class in File Compiled With Ant

javaeclipseantcsvjar

提问by Justian Meyer

I compile and run my program in Eclipse and everything works fine, but when I package it with Ant and run it, I get this error:

我在 Eclipse 中编译并运行我的程序,一切正常,但是当我用 Ant 打包并运行它时,出现此错误:

Exception in thread "main" java.lang.NoClassDefFoundError: org/supercsv/io/ICsvB
eanReader
Caused by: java.lang.ClassNotFoundException: org.supercsv.io.ICsvBeanReader
        at java.net.URLClassLoader.run(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        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)
Could not find the main class: jab.jm.main.Test. Program will exit.

Note that this is a runtimeerror and not a compiler errorwith Ant.

请注意,这是一个运行时错误,而不是Ant的编译器错误

I've built this project in the past with 0 issues and now it suddenly acts up on me when I add a second package to my lib folder?

我过去用 0 个问题构建了这个项目,现在当我将第二个包添加到我的 lib 文件夹时它突然对我起作用

Here's the build file for reference:

这是构建文件以供参考:

<?xml version="1.0" ?>

<project name="ServerJar" default="dist" basedir=".">
 <description>
  Builds client files into .jar
 </description>
 <!-- [build variables] -->
 <property name="src" location="src" />
 <property name="build" location="build" />
 <property name="dist" location="dist" />
 <property name="lib" location="lib" />
 <!-- [path to packages] -->
 <path id="master-classpath">
     <fileset dir="${lib}">
         <include name="*.jar"/>
     </fileset>
 </path>


 <target name="init">
  <!-- makes time stamp to be used in jar name -->
  <tstamp />
  <!-- creates build directory structure -->
  <mkdir dir="${build}" />
 </target>

 <target name="compile" depends="init" description="Compiles the source">
  <!-- compiles the java code from ${src} into ${build} -->
  <!-- <javac srcdir="${src}" destdir="${build}" /> -->
  <javac destdir= "${build}">
      <src path="${src}"/>
      <classpath refid="master-classpath"/>
  </javac>
 </target>

 <target name="dist" depends="compile" description="Generates distributable">
  <!-- creates the distribution directory -->
  <mkdir dir="${dist}/lib" />

  <!-- puts everything in ${build} into the jar file -->
  <jar jarfile="${dist}/lib/CC-${DSTAMP}.jar" basedir="${build}">
   <manifest>
    <attribute name="Main-Class" value="jab.jm.main.Test" />
   </manifest>
  </jar>

  <!-- makes a jar file for quick test execution -->
  <jar jarfile="${dist}/lib/CC.jar" basedir="${build}">
   <manifest>
    <attribute name="Main-Class" value="jab.jm.main.Test" />
   </manifest>
  </jar>
 </target>

 <target name="clean" description="Cleans up the extra build files">
  <!-- deletes the ${build} and ${dist} directories -->
  <delete dir="${build}" />
  <delete dir="${dist}" />
 </target>
</project>

Thanks in advance for the help!

在此先感谢您的帮助!

EDIT:

编辑:

Here's what the construction for my main class looks like (this is not the actual file, but this is what I based mine on). The construction is very odd for a java program and might be giving Ant some issues. Any recommendations on how to reconstruct this? I got a bunch of errors when trying to separate this into multiple parts. I've just never seen a construction like this before (yes, I understand HOW it works (and it does when compiled), but Ant might not like it).

这是我的主类的构造(这不是实际文件,但这是我的基础)。Java 程序的构造非常奇怪,可能会给 Ant 带来一些问题。关于如何重建这个的任何建议?尝试将其分成多个部分时,我遇到了一堆错误。我以前从未见过这样的结构(是的,我理解它是如何工作的(编译时也是如此),但 Ant 可能不喜欢它)。

import java.io.FileReader;
import java.io.IOException;

import org.supercsv.cellprocessor.Optional;
import org.supercsv.cellprocessor.ParseDate;
import org.supercsv.cellprocessor.ParseInt;
import org.supercsv.cellprocessor.constraint.StrMinMax;
import org.supercsv.cellprocessor.constraint.Unique;
import org.supercsv.cellprocessor.ift.CellProcessor;
import org.supercsv.io.CsvBeanReader;
import org.supercsv.io.ICsvBeanReader;
import org.supercsv.prefs.CsvPreference;

class ReadingObjects {
    static final CellProcessor[] userProcessors = new CellProcessor[] {
        new Unique(new StrMinMax(5, 20)),
        new StrMinMax(8, 35),
        new ParseDate("dd/MM/yyyy"),
        new Optional(new ParseInt()),
        null
    };

    public static void main(String[] args) throws Exception {
        ICsvBeanReader inFile = new CsvBeanReader(new FileReader("foo.csv"), CsvPreference.EXCEL_PREFERENCE);
        try {
          final String[] header = inFile.getCSVHeader(true);
          UserBean user;
          while( (user = inFile.read(UserBean.class, header, userProcessors)) != null) {
            System.out.println(user.getZip());
          }
        } finally {
          inFile.close();
        }
   }
}

public class UserBean {
    String username, password, town;
    Date date;
    int zip;

    public Date getDate() {
        return date;
    }
    public String getPassword() {
        return password;
    }
    public String getTown() {
        return town;
    }
    public String getUsername() {
        return username;
    }
    public int getZip() {
        return zip;
    }
    public void setDate(final Date date) {
        this.date = date;
    }
    public void setPassword(final String password) {
        this.password = password;
    }

    public void setTown(final String town) {
        this.town = town;
    }
    public void setUsername(final String username) {
        this.username = username;
    }
    public void setZip(final int zip) {
        this.zip = zip;
    }
}

Notice how the class's name is actually UserBean and it contains a non-public class named ReadingObjects within it that holds the main method.

请注意该类的名称实际上是 UserBean 并且它包含一个名为 ReadingObjects 的非公共类,该类包含 main 方法。

回答by Mark O'Connor

Looks like your runtime classpath is missing the jar containing the class org.supercsv.io.ICsvBeanReader.

看起来您的运行时类路径缺少包含org.supercsv.io.ICsvBeanReader类的jar

The gotchais that you cannot set the classpath from the command-line when calling an executable jar. You have to set it within the manifest as follows:

疑难杂症的是,调用一个可执行的JAR,当你不能从命令行设置类路径。您必须在清单中设置它,如下所示:

<target name="dist" depends="compile" description="Generates distributable">
    <!-- creates the distribution directory -->
    <mkdir dir="${dist}/lib" />

    <!-- Remove manifest. This jar will end up on the classpath of CC.jar -->
    <jar jarfile="${dist}/lib/CC-${DSTAMP}.jar" basedir="${build}"/>

    <!-- Fancy task that takes the pain out creating properly formatted manifest value -->
    <manifestclasspath property="mf.classpath" jarfile="${dist}/lib/CC.jar">
        <classpath>
            <fileset dir="${dist}/lib" includes="*.jar"/>
        </classpath><!--end tag-->
    </manifestclasspath>

    <!-- This is the executable jar -->
    <jar jarfile="${dist}/lib/CC.jar" basedir="${build}">
        <manifest>
            <attribute name="Main-Class" value="jab.jm.main.Test"/>
            <attribute name="Class-Path" value="${mf.classpath}"/> 
        </manifest>
    </jar>

</target>

This approach will allow you to run the jar as follows:

这种方法将允许您按如下方式运行 jar:

java -jar CC.jar

Without the extra manifest entry you have to run the jar as follows:

如果没有额外的清单条目,您必须按如下方式运行 jar:

java -cp CC.jar:CC-DSTAMPVALUE.jar jab.jm.main.Test

Note

笔记

Only the CC.jar is executable and needs the special manifest. Using this pattern means future additional jars, placed into the lib directory, will be automatically included in the run-time classpath. (Useful for open source dependencies like log4j)

只有 CC.jar 是可执行的并且需要特殊的清单。使用此模式意味着将来放置在 lib 目录中的其他 jar 将自动包含在运行时类路径中。(对于像 log4j 这样的开源依赖很有用)

Obviously, when running the CC.jar you'll get a similar error if the jar files are not present :-)

显然,在运行 CC.jar 时,如果 jar 文件不存在,您将收到类似的错误:-)

回答by Tom Tresansky

Have you tried explicitly specifying the classpath when running the jar, in order to ensure the new library is on it?

您是否尝试过在运行 jar 时显式指定类路径,以确保新库在其上?

Perhaps library 1 is present in the default classpath, so your project ran fine until you added library 2, which was not present. When running within Eclipse, the IDE may be automatically adding library 2 to the classpath for you. You can check the classpath in your project's Run Configuration within Eclipse, and make sure you are including everything there when not running via the IDE.

也许库 1 存在于默认类路径中,因此您的项目运行良好,直到您添加了库 2,而该库不存在。在 Eclipse 中运行时,IDE 可能会自动将库 2 添加到类路径中。您可以在 Eclipse 中检查项目运行配置中的类路径,并确保在不通过 IDE 运行时包含所有内容。

回答by GuruKulki

This might be happening because of the location of the generated class files. ie when you build a through eclipse it generates the class files in the location which is specified as the Output folder for ex: bin, and while running it looks at this location for the class files.

这可能是由于生成的类文件的位置而发生的。即,当您通过 Eclipse 构建时,它会在指定为输出文件夹的位置生成类文件,例如:bin,并在运行时查看此位置的类文件。

So check whether your ant is generating the class files at the same location as the output folder mentioned in the BuildPath configuration. If not change the output folder location to the location where your ant is generating the class files.

因此,请检查您的 ant 是否在与 BuildPath 配置中提到的输出文件夹相同的位置生成类文件。如果不是,请将输出文件夹位置更改为您的 ant 生成类文件的位置