22

我编写了以下 Java 源文件 ( Hello.java):

package com;

public class Hello {
    public static void main(String[] args) {
        System.out.println("Hello!");
    }
}

我把这个保存到C:/tmpjava/Hello.java.

从命令行,我导航到该目录并运行javac Hello.java. 然后我运行dir

  • Hello.class
  • Hello.java

然后,从我刚刚运行的同一个目录中javac,我运行java Hello.class并得到:

Exception in thread "main" java.lang.NoClassDefFoundError: Hello/class
Caused by: java.lang.ClassNotFoundException: Hello.class
    at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:306)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:301)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:247)
Could not find the main class: Hello.class.  Program will exit.

这里发生了什么?!?怎么javac能跑得好,但不行java

4

5 回答 5

50

你的类Hello属于包com。因此,您的班级的完全限定名称是com.Hello. 当您在命令行上使用 java 调用程序时,您应该提供包含您的main方法的类的完全限定类名并省略.class,如下所示:

java com.Hello

java 程序需要这个完全限定的类名来理解你指的是哪个类。

但是你还有另一个问题。java 程序使用文件系统定位包、子包和属于它们的类。因此,如果您有一个类似的包结构com.Hello,java 程序期望在名为com的目录中找到一个名为Hello.class的类文件,如下所示:com/Hello.class。事实上,你可以在你所看到的中观察到这种行为;您错误地使用了Hello.class,java 将其解释为名为 的包名为的,并且正在寻找目录结构Hello/classExceptionHelloclass

java.lang.NoClassDefFoundError: 你好/类

但是编译器 javac默认设置这个目录结构。请参阅javac 的文档,但重要的是:当您进行编译时,您可以使用-d标志指定目标目录:

-d 目录

设置类文件的目标目录。目标目录必须已经存在;javac 不会创建目标目录。如果类是包的一部分,javac 会将类文件放在反映包名称的子目录中,并根据需要创建目录。例如,如果您指定 -dc:\myclasses 并且类名为 com.mypackage.MyClass,则类文件名为 c:\myclasses\com\mypackage\MyClass.class。

如果未指定 -d,javac 会将类文件放在与源文件相同的目录中。

粗体字的最后一点对于初学者来说是很多困惑的根源,并且是您自己问题的一部分。

所以你有两种选择:

  1. 在您的情况下,如果您提供当前目录作为目标目录就可以了,就像这样(句点.表示当前目录):

    javac -d . Hello.java
    

    如果您像这样调用编译器,它将为您创建com目录,并将您编译的类文件放入其中,这是 java 程序期望找到它的方式。然后,当您从c:\tmpJava像上面那样运行 java 时,您的程序应该执行。

  2. 您可以使用反映包结构的目录结构设置源代码:将源文件Hello.java放在名为com的目录中,在您的情况下为:c:\tmpJava\com\Hello.java。现在,从c:\tmpJava你可以像这样运行你的 javac 编译:

    javac com\Hello.java
    

    您没有提供-d标志,但这很好,因为您自己创建了目录结构,并再次引用上面的文档:

    如果未指定 -d,javac 会将类文件放在与源文件相同的目录中。

    同样,当您像上面那样运行 java 时,您的程序应该执行。

    请注意,第二种选择是 java 程序员常用的一种选择:源代码文件组织在一个目录结构中,该目录结构反映了包结构。

在这个解释中,我们忽略了类路径的概念。您还需要了解编写 java 程序,但是在您只是在当前目录中编译程序的情况下 - 如果您在编译类时遵循上述两种选择之一 - 您可以在不设置类路径的情况下逃脱,因为,默认情况下,java 程序将当前目录作为类路径。另一个引用,这是来自java 文档的引用:

-cp 类路径

指定目录、JAR 档案和 ZIP 档案的列表以搜索类文件。类路径条目由分号 (;) 分隔。指定 -classpath 或 -cp 会覆盖 CLASSPATH 环境变量的任何设置。

如果未使用 -classpath 和 -cp 且未设置 CLASSPATH,则用户类路径由当前目录 (.) 组成。

请注意,当您使用像 Eclipse 这样的 IDE 来运行您的 java 代码时,这主要是为您处理的,但您仍然会遇到类路径问题。

于 2012-08-23T16:29:31.037 回答
14

Java 命令的语法是:

java [classname]

不是

java [filename]

Java 在其类路径中查找您提供的名称的类。通常类路径包括当前目录,所以:

java Hello

... 将在当前目录中找到 Hello.class,并运行其 main() 方法。

但是,如果该类位于其他位置(例如 .jar 或文件系统中的其他位置),您可以使用 CLASSPATH 环境变量或在命令行中指定它:

java -cp build/classes Hello
java -cp build/jars/myjar.jar Hello
于 2012-08-23T16:24:43.630 回答
4

该类应该在C:\tmpjava\com\Hello.class
并且您应该从以下位置运行C:\tmpjavajava -cp . com.Hello
当您将一个类放入包中时,它定义了该类的文件结构。即你的包中的类com应该在文件夹中com

于 2012-08-23T16:19:28.003 回答
2

运行以下命令

java -cp . com.Hello
于 2012-08-23T16:20:25.207 回答
1

java -classpath c:\tmpjava com.Hello

于 2012-08-23T16:22:27.217 回答