java 虚拟机通过 classloader来装载字节码。classloader 从哪里知道 java 本身的类库及用户自己的类在什么地方呢?
实际上 java 虚拟机是由 java luncher 初始化的, 也就是 java (或 java.exe)这个程序来做的。虚拟机按以下顺序搜索并装载所有需要的类:
1、引导类: 组成 java 平台的类,包含 rt.jar 和 i18n.jar 中的类。
2、扩展类: 使用 java 扩展机制的类,都是位于扩展目录($JAVA_HOME/jre/lib/ext)
中的 .jar包。
3、用户类: 开发者定义的类或者没有使用 java 扩展机制的第三方产品。你必须在
命令行中使用 -classpath 选项或者使用 CLASSPATH 环境变量来确定这些类的位置
其中,用户类路径就是一些包含类字节码文件的目录,我们需要显式或隐式去指定,可以使用以下几种方式指定
• 默认情况下,编译器和 JVM 在当前目录查找所需的源文件和类文件。
• CLASSPATH 环境变量,一旦设置将缺省值覆盖。
• 命令行参数 -cp 或者 –classpath, 一旦指定将上两者覆盖。
• 由 -jar 参数指定的 .jar包,就把所有其他的值覆盖,所有的类都来自这个指定的jar包中。
如,类文件不在JVM当前运行目录,且不在CLASSPATH路径下时
[root@localhost test]# cd ..
[root@localhost ~]# java Clothes
Error: Could not find or load main class Clothes
指定类文件的全路径或相对于JVM运行目录的路径,如
[root@localhost test]# java test/Clothes
Error: Could not find or load main class test.Clothes
可以看到运行的时候,虽然指定了.class文件的准确位置,但JVM将后面的参数看做是test.Clothes类,显然我们没有这个类
上述原因都是因为JVM当前所在的目录下没有Clothes.class文件。
这时我们可以使用第二种方式,即使用-classpath (或-cp)选项指定查找路径,如
[root@localhost ~]# java -classpath /test/ Clothes
i'm making a XL size cloth with color red
i'm making a pair of shoes with color red and size 41
[root@localhost ~]#
cp的意思就是class search path of directories and zip/jar files。
我们可以不必每次都指定路径,而是使用classpath环境变量。
类目录的绝对路径叫做 classpath,设置在系统变量 CLASSPATH 中。
环境变量CLASSPATH的值可以通过操作系统来设置,通常CLASSPATH包含一个或多个目录,用作查找.class文件的根目录。
[root@localhost ~]# env | grep CLASSPATH
CLASSPATH=.:/usr/java/jdk1.8.0_181/lib:/usr/java/jdk1.8.0_181/jre/lib:
[root@localhost ~]#
假如我们将Clothes.class文件移动到/usr/java/jdk1.8.0_181/jre/lib/路径下,则可以在任何目录下运行这个类文件,如
[root@localhost ~]# mv /test/Clothes.class /usr/java/jdk1.8.0_181/jre/lib/
[root@localhost ~]# java Clothes
i'm making a XL size cloth with color red
[root@localhost ~]#
再次说明编译的时候和不需要和CLASSPATH打交道,只要在javac命令后面指明了.java文件的位置即可,
[root@localhost ~]# javac /test/Clothes.java
[root@localhost ~]# ll /test/Clothes.class
-rw-r--r-- 1 root root 771 Jun 20 17:19 /test/Clothes.class
但是在执行.class文件的时候,必须要进到.class所在的路径或使用-cp指定类文件所在的相对或绝对路径,或者在任意一个目录下执行,但需要事先将.class的包所在的路径添加到CLASSPATH。否则都会报错:Error: Could not find or load main class XXX。