Java 源程序的编译大家都知道,也就是 cmd 中到源文件所在目录下 javac **.java
即可,当程序中有包声明还能简简单单的直接 javac **.java
吗?答案当然是 no,下面举个简单的例子证明一下直接 javac **.java
会怎么样。
如下:F:\javaweb2班\20160531目录下有A.java 文件,注意源文件中有包声明:
实例
现在对 A.java 直接进行编译 javac A.java 结果如下:
我们可以清楚地看到当前目录下出现了A.class字节码文件,就这样能不能直接运行呢?试试吧!
出现了这样的错误,原因是我们在 A.java 中有包声明,当执行 A.java 源文件时 java 虚拟机首先会在 A.java 当前目录寻找字节码文件,虽然这下找到了,但是因为在 A.java 中有包声明,java 虚拟机紧接着去包目录下寻找有没有 A.class 字节码文件,这下找到了才能够成功执行,不信我们做做实验!(这里我再教教大家怎么编译有包声明的 A.java,也就是打包编译 javac -d . A.java 即可)
这下就打包编译成功了,这时我们可以看到 mypack 目录下出现了 A.class 文件,下面试试执行吧
这时我们发现执行还是不能通过??这时为什么呢?这里是许多新手遇到的问题,在这里大家必须谨记一点:Java 命令的参数是"类的完整类名",而不是"文件名"。
这个源程序的完整类名应该是 mypack.A,因此应该这么写:java mypack.A
这下就可以成功执行了!
别急!!下面还有更精彩的!作为包的使用怎么可能会不讲解一下包的导入创建编译呢!
这次我们引入另外一个测试类 Test.java,代码如下:
实例
编译执行如下:
结果意料之中肯定是能够编译执行的,执行流程是这样的:Test.java 编译之后生成的字节码文件在当前目录(编译时他会寻找 mypack 中是否有 A.class 文件,若无,编译不通过),执行时,由于 Test.java 中没有包声明,java 虚拟机首先在当前目录找到 Test.class 就会执行,执行到程序中引用到A类的地方,然后 java 虚拟机在当前目录看是否有 A.class 字节码文件,这时即使发现了也会根据源程序中的包导入进入包中寻找 A.class,找到才能执行成功(其实在编译阶段就已经寻找了!)
下面更进一步:若我们给测试类 Test.java 加上包声明 package mypack1;
这时我们对 Test.java 进行打包编译,这里我又要给大家讲解两个知识点:1.打包编译时,会自动创建包目录,不需要自己新建包名文件夹;2.当当前目录有多个java文件需要编译或打包编译时,javac -d . *.java
指令可以给当前目录下的所有 java 文件根据程序中是否有包声明进行编译或打包编译。
这时我们又该如何执行 Test.java 文件呢?java Test.java 吗??显然这样是不行的,还记得我前面讲过的么:Java 命令的参数是"类的完整类名",而不是"文件名"。
因此我们需要这样执行:
这样就:
上面讲的这些都是一般情况,也就是类路径都是在当前目录下,当类路径不在当前目录下是否还能执行呢?又该如果执行呢?
如下图我把 Test.java 放到外面一层目录,这时我们就需要自己设置 classpath 参数。例如:F:\javaweb2班>java -cp F:/javaweb2班/20160531 mypack1.java;或者在任意目录下:java -cp F:/javaweb2班/20160531 mypack1.java
这样就成功了!具体的执行流程大家自己分析体会吧!
总结一下
-
1、Java 命令的参数是"类的完整类名",而不是"文件名"。
- 2、打包编译时,会自动创建包目录,不需要自己新建包名文件夹。
-
3、当当前目录有多个 java 文件需要编译或打包编译时,
javac -d . *.java
指令可以给当前目录下的所有 java 文件根据程序中是否有包声明进行编译或打包编译。 -
4、当类路径不在当前目录下时,需要用到
java -cp ...
,如:java -cp F:/javaweb2班/20160531 mypack1.java。 - 5、要清楚 java 虚拟机根据包声明包导入执行字节码文件的流程。