当我们试图在命令行编译运行一个简单的java程序HelloWorld.java,编译命令javac HelloWorld.java顺利通过,但是执行运行java HelloWorld却报出:“错误:找不到或无法加载主类 xxx(Error: Could not find or load main class )”,注:Java 1.7以下是这样的Exception in thread "main" java.lang.NoClassDefFoundError: HelloWorld.
为了解决这个问题,就必须知道java是寻找和加载类的原理。首先有个环境变量叫做CLASSPATH,它包含所有Java类文件的目录。如果它找不到我们所运行程序的main方法就会抛出上面的错误。在Eclipse里面编译运行程序很简单,因为它把Classpath设置好了,但是在命令行,你就得自己处理了。CLASSPATH和PATH不同,PATH用于定位系统可执行文件或命令,在Windows系统是扩展名为 .exe, .dll的文件,在Linux系统是 .so文件。Path也用来定位java程序的本地库。CLASSPATH是用来定位类文件或JAR文件,java类加载器通过CLASSPATH来查找和加载类。
下面用一个例子来演示如何在命令行执行java程序。
项目目录E:\Users\workspace,项目名hellojava,包名com.sommer.learn,类名HelloJava
package com.sommer.learn; public class HelloJava { public static void main(String[] args) { System.out.println("Hello Java!"); } }
提示出错,因为这个类的全限定全名是com.sommer.learn.HelloJava,而不是HelloJava。重新输入命令 java com.sommer.learn.HelloJava
为什么还是出错呢?因为我们没有CLASSPATH环境变量,也没有用java命令的-classpath或-cp选项,所以Java默认在当前目录下寻找主类 com/sommer/learn/HelloJava.class,但是我们已经在com/sommer/learn目录里面了,所以找不到类。将目录切换到包的父路径,再来运行一遍
这回终于可以了!
如果想从任意路径来运行程序该怎么办呢?如上文提到的,使用java命令的-classpath或-cp选项,或设置CLASSPATH。先来看第一种
第二种设置CLASSPATH
windows+pause快捷键打开系统属性——高级系统设置——环境变量,在系统变量里新建CLASSPATH(不区分大小写),一般设变量值为 .;%JAVA_HOME%\lib;%JAVA_HOME%\lib\tools.jar,注意最前面有个点号,表示当前路径。
可以在命令行输入echo %CLASSPATH%来查看变量值。如果将本项目E:\Users\workspace\hellojava\bin加在classpath变量值里(用分号跟其他的隔开),则可以在任意路径执行命令 java com.sommer.learn.HelloJava。
但是一般不建议这样设,如果把每个项目的类路径都添加到系统CLASSPATH会造成混乱,建议用java命令的 -classpath选项来指定。
如果在eclipse里运行一个java类时报出“错误:找不到或无法加载主类 xxx”这样的错,一般是项目没有编译,检查菜单栏的Project——Build Automaticlly有没有被勾上。