cmd编译运行java程序

cmd编译运行java程序

第一节:配置Java运行时环境

本节以Windows XP为例,来配置Java运行环境。这节很简单,我就长话短说。

1:下载JDK开发包,解压安装。
2:在用户环境变量中增加PATH变量(在系统的环境变量中增加也行,但是可能需要重新启动操作系统),并设置其值。

例如我把JDK安装在F:\program\java目录下,则把PATH变量设置成F:\program\java\jdk\bin,因为这个路径下有我们要编译、运行Java程序所需要的所有工具,比如javac.exe,java.exe,jar.exe....。之后我们就可以在cmd下使用javacjavajar等命令了。

例如有一下Java源文件Test.java,执行javac Test.java就会在当前目录下生成Test.class文件。之后我们运行java Test即可运行程序。当然也可以人为指定class文件的生成位置,利用-d选项。比如 javac -d D:\ Test.java,那么就可以在D:\目录下看到Test.class了。

第二节:代码的编译以及运行

对于写Java程序而言,我们主张源文件与class文件都有相同的目录结构,这样不仅会方便代码的组织管理,也使得在编译以及运行Java程序时少犯错误。

1:位于默认包下的文件的编译以及运行

//D:\src\A.java
class A{

    private int value;
    
    public A(){
        value = 0;
    }
    
    public void setValue(int var){
        value = var;
    }
    
    public int getValue(){
        return value;
    }

}

//D:\src\Test.java
public class Test{
    public static void main(String[] args){
        A a = new A();
        a.setValue(10);
        System.out.println(a.getValue());
    }
}

我们运行javac Test.java,会发现D:\src目录下多了A.classTest.class两个文件。运行java Test则可以在Dos中看到输出为10。我们发现这两个Java文件中并没有package语句,所以这两个文件同属于一个默认包中(没有显示写package的文件都是属于默认包的)

2: Java类发现规则

要想成功编译java文件,需要有classpath和包名的共同配合。

类发现规则:class文件所在目录 = classpath + '\' + 包名中的'.'全变成'\'

这个规则大家一定要记住,保证万事OKJava就是靠这个规则来寻找需要的class文件的。

例如:classpath=D:

//D:\src\A.java
package src;
public class A{

    private int value;
    
    public A(){
        value = 0;
    }
    
    public void setValue(int var){
        value = var;
    }
    
    public int getValue(){
        return value;
    }

}

//D:\Test.java
import src.*;
public class Test{
    public static void main(String[] args){
        A a = new A();
        a.setValue(10);
        System.out.println(a.getValue());
    }
}

如果想要javac Test.java成功,则需要在D:\src(D:\src = D:\ + src)路径下能够找到A.class,否则编译出错。如果不存在D:\src目录,则会返回包名不存在的错误信息。如果D:\src存在,但是该路径下不存在A.class时,则报告没有找到class文件。如果java文件存放目录正确,并且也正确的指定了其所在的包,则javac会自动编译A.java来生成A.class

例如:上例就能够正确的生成所有需要的class文件。但是如果把A.javapackage语句去掉。之后编译A.java生成A.class文件,之后在D:\下运行javac Test.java并不会成功,虽然在D:\src下存在A.class文件。

所以综上所述,package语句中包名必须与生成的class文件所在的目录保持一个关系,使得javac能够找到相应的class文件并且验证一个文件是否处于正确的包中。

那么classpath是什么呢?从名字也能猜得到,classpath代表class文件的存放路径。这是个环境变量,可以添加到用户的环境变量中,其值则根据编程时具体包名、以及类的存放路径来设定。如果不想用环境变量,也可在编译时用-classpath来设置。

classpath可以设置多个路径,路径之间利用';'相隔。编译时,javac会逐个尝试classpath的每个路径和文件的包名的组合,直到找到相应的class文件为止。但是如果通过classpath与包名的组合在多个不同的路径下都找到了同名类,则会返回错误信息。

由于javac默认情况下在java文件的当前目录生成class文件,所以人们经常把'.'添加到classpath中,代表当前文件所在的路径。高版本的Java编译以及运行程序可能会自动把'.'添加到classpath中。上面的例子中由于Test.java位于A.java的上一级目录中,所以把classpath设置成'.'时,也能编译成功。

现在说说把classpath设置成'.',它代表当前运行目录。那么什么是这个当前运行目录呢?其实就是运行javac时所在的目录,注意:切不可认为是当前编译的那个java文件所在的目录。

如果java文件中显示的写了package packagename 则为class文件指定了包,那么这个class文件也必须应在packagename转换后的相应子目录下。如果不写,则表示该class文件在默认包中。

现在来说说如运行java程序。
例如:classpath=D:

/D:\src\A.java
package src;
public class A{

    private int value;
    
    public A(){
        value = 0;
    }
    
    public void setValue(int var){
        value = var;
    }
    
    public int getValue(){
        return value;
    }

}

//D:\src\Test.java
package src;
import src.*;
public class Test{
    public static void main(String[] args){
        A a = new A();
        a.setValue(10);
        System.out.println(a.getValue());
    }
} 

我们如何运行这个程序呢?

首先在src目录下运行: javac Test.java

之后在任意目录下运行:java  src.Test

注意:切不可运行java Test命令,会提示找不到类的错误(即使在Test.class文件所在目录也不行)。所以运行指定了package程序时,需要显示的写出其包名,即:java packagename.classname,其遵循与class文件一样的寻找规则,即在classpath+packagename'.'变成'\' 所表示的路径下能够找到相应的classname所指的class文件。

第三节:如何打jar包以及使用jar

jar包使用jar命令。

//D:\src\A.java
package src;
public class A{

    private int value;
    
    public A(){
        value = 0;
    }
    
    public void setValue(int var){
        value = var;
    }
    
    public int getValue(){
        return value;
    }

}

首先在src目录下运行javac A.java生成A.class。之后在D:\目录下运行jar -cvf test.jar src就会在D:\这个目录下生成test.jar,即我们打的jar包。那么我们如何使用这个jar包呢?首先必须把test.jar的路径加入到classpath中。之后我们有如下的测试程序。

import src.*; //必须得有此import,因为A.class位于src包中。如果不写则编译不通过。

public class B{

    public static void main(String[] args){
        A a = new A();
    }
}

jar文件其实是一个压缩包,可以很容易的用解压软件进行解压。解压后有如下目录关系:

test.jar
   |--src---|--A.java
            |--A.class

这里可以把test.jar当成目录来处理,javac会自动在classpath中解压这个test.jar并根据B.java中的import找到其中的src文件夹下的A.class文件并使用。

现在看看我们的打包命令 jar -cvf test.jar src  //意思是把src文件夹整个打包

如果我们这么打包: 在src目录下运行jar -cvf test.jar A.java A.class,行吗?答案是不行。因为这样打包后路径关系为:

test.jar
   |--A.java
   |--A.class

javac寻找A.class时发现test.jar下不存在src目录,则会报不存在包的错误。所以大家一定要注意打包时应该把那个目录打进去,不应把哪个目录打进去。如果对属于默认包的class文件打包,则只要把jar包路径添加到classpath中即可使用,无需再在程序中使用import。当然jar命令还有很多参数可选,来满足各种不同需要的打包需求,这些就不说了。

4:利用Makefile自动编译运行Java程序

下面给出一个简单的例子:这个例子中,没有手动设置Windows的环境变量,而是用-classpath指定,这样方便在不同机器之间移植

如下目录结构:

project---|---src---|---main---|---Test.java
          |         |---package1---|---A.java
          |         |              |---B.java
          |         |
          |         |---package2---|---C.java
          |         |---Makefile
          |
          |---classes---|---main---|---Test.class
                        |---package1---|---A.class
                        |              |---B.class
                        |
                        |---package2---|---C.class

代码如下:

//A.java
package package1;
public class A{

    private int value;
    
    public A(){ value = 0; }
    
    public void setValue(int var){
        value = var;
    }
    
    public int getValue(){
        return value;
    }
}

//B.java
package package1;
public class B{

    private String name;
    
    public B(){ name = null; }
    
    public void setName(String name){
        this.name = name;
    }
    
    public String getName(){
        return name;
    }

}

//C.java
package package2;
public class C{

    private String name;
    
    public C(){ name = null; }
    
    public void setName(String name){
        this.name = name;
    }
    
    public String getName(){
        return name;
    }

}

//Test.java
import java.util.*;
import package1.*;
import package2.*;
public class Test{

    public static void main(String[] args){

        A a = new A();
        B b = new B();
        C c = new C();
        
        a.setValue(3);
        b.setName("Java");
        c.setName("Test");
        
        System.out.println(a.getValue());
        System.out.println(b.getName());
        System.out.println(c.getName());
    }
}

Makefile如下:

//Makefile
target:
    javac ./package1/A.java -d ../classes
    javac ./package1/B.java -d ../classes
    javac ./package2/C.java -d ../classes
    javac ./main/Test.java  -d ../classes/main/ -classpath ../classes
run:
    java -classpath ../classes/main;../classes Test
clean:
    rm -rf ../classes/package1/*.class
    rm -rf ../classes/package2/*.class
    rm -rf ../classes/main/*.class

运行:make
D:\Java\project\src>make
javac ./package1/A.java -d ../classes
javac ./package1/B.java -d ../classes
javac ./package2/C.java -d ../classes
javac ./main/Test.java -d ../classes/main/ -classpath ../classes

运行:make run
D:\Java\project\src>make run
java -classpath ../classes/main;../classes Test
3
Java
Test

这样,其实我们完全可以不用EclipseNetbeans等开发工具就能自己编译运行自己的系统了。


你可能感兴趣的:(java)