Java无法找到主类
我编写了以下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
?
你的类Hello
属于包com
。 因此,您的类的完全限定名称是com.Hello
。 在命令行上使用java调用程序时,应该提供包含main
方法的类的完全限定类名,并省略.class ,如下所示:
java com.Hello
java程序需要这个完全限定的类名来理解你所引用的类。
但是你有另一个问题。 java程序使用文件系统定位包,子包和属于它们的类。 因此,如果你有一个像com.Hello
这样的包结构,那么java程序希望在一个名为com的目录中找到一个名为Hello.class的类文件,如下所示: com / Hello.class 。 实际上,您可以在您看到的Exception
中观察到此行为; 你错误地使用了Hello.class ,java解释为一个名为Hello
的包 ,一个名为class
,并且正在寻找目录结构Hello / class :
java.lang.NoClassDefFoundError:Hello / class
但是编译器javac默认情况下不会设置此目录结构。 请参阅javac的文档 ,但重要的是:编译时,可以使用-d
标志指定目标目录:
-d目录
设置类文件的目标目录。 目标目录必须已存在; javac不会创建目标目录。 如果类是包的一部分,则javac将类文件放在反映包名的子目录中,根据需要创建目录。 例如,如果指定-dc:\ myclasses并且该类名为com.mypackage.MyClass,则类文件名为c:\ myclasses \ com \ mypackage \ MyClass.class。
如果未指定-d,则javac将类文件放在与源文件相同的目录中。
粗体的最后一点是初学者很多混乱的根源,也是你自己问题的一部分。
所以你有两个选择:
-
在你的情况下,如果你提供当前目录作为目标目录,就好了(句号
.
表示当前目录 ):javac -d . Hello.java
如果您像这样调用编译器,它将为您创建com目录,并将编译后的类文件放入其中,这是java程序期望找到它的方式。 然后当你从c:\ tmpJava运行java时,你的程序应该执行。
-
您可以使用镜像包结构的目录结构来设置源代码:将源文件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代码时,这主要是为您处理的,但您仍会遇到类路径问题。
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
该类应该在C:\tmpjava\com\Hello.class
你应该从C:\tmpjava
: java -cp . com.Hello
java -cp . com.Hello
将类放入包中时,它定义了类的文件结构。 也就是你的类在com
包中应该在文件夹com
运行以下命令
java -cp . com.Hello
java -classpath c:\ tmpjava com.Hello
- ClassNotFoundException oracle.jdbc.driver.OracleDriver仅在servlet中使用Eclipse
- 加载JDBC驱动程序中的classNotFoundException
- 在代码上获取ClassNotFoundException:“Class.forName(”com.microsoft.sqlserver.jdbc.SqlServerDriver“);”
- oracle.jdbc.driver.OracleDriver ClassNotFoundException
- ClassNotFoundException在修改后的SimpleShortestPathsVertex上运行GiraphRunner
- Tomcat为其他开放日食项目中的类抛出ClassNotFoundexception
- java.lang.ClassNotFoundException:org.springframework.http.converter.json.MappingJackson2HttpMessageConverter
- 启动java.util.concurrent.ExecutionException期间子容器失败
- 没有在路径DexPathList上找到类