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

时间:2010-06-29 18:22:09

标签: java eclipse ant csv jar

我在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$1.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.

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

我过去已经构建了这个项目,有0个问题,现在当我向我的lib文件夹中添加第二个包时,它会突然出现在我身上

这是构建文件供参考:

<?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>

提前感谢您的帮助!

编辑:

这是我的主类的构造看起来像(这不是实际的文件,但这是我的基础)。对于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;
    }
}

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

3 个答案:

答案 0 :(得分:12)

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

gotcha 是调用可执行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>

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

java -jar CC.jar

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

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

注意

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

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

答案 1 :(得分:0)

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

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

答案 2 :(得分:0)

由于生成的类文件的位置,可能会发生这种情况。即,当你构建一个通过eclipse时,它会在指定为ex:bin的Output文件夹的位置生成类文件,并在运行时查看该文件的这个位置。

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