我在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方法。
答案 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配置中提到的输出文件夹相同的位置生成类文件。如果不是将输出文件夹位置更改为您的蚂蚁生成类文件的位置。