
1. Java程序的标准编译与执行流程
在深入探讨makefile的应用之前,理解java程序的标准运行方式至关重要。与直接运行.java源文件不同,java程序通常需要先编译成字节码文件(.class),然后再由java虚拟机(jvm)执行。
标准流程包括两个主要步骤:
-
编译 (Compilation): 使用javac命令将Java源代码文件(例如AvlTree.java)编译成对应的字节码文件(AvlTree.class)。
javac AvlTree.java
执行此命令后,会在当前目录下生成AvlTree.class文件。
-
执行 (Execution): 使用java命令运行已编译的字节码文件。需要注意的是,java命令后跟随的是类的完整名称(不带.class扩展名),并且通常需要指定类路径(classpath)。
立即学习“Java免费学习笔记(深入)”;
java -cp . AvlTree inputs.txt
- -cp .:指定当前目录(.)为类路径,这样JVM才能找到AvlTree.class。
- AvlTree:要执行的主类名。
- inputs.txt:这是传递给Java程序的命令行参数,可以在Java代码中通过main方法的String[] args参数获取。
重要提示: 直接运行java AvlTree.java inputs.txt虽然在某些JDK版本中可能奏效,但它实际上是JDK的一个便利功能,在后台隐式地进行了编译,并非Java程序的“正常”或推荐的执行方式。标准做法是先显式编译,再执行编译后的类文件。
2. Makefile在Java项目中的应用
尽管对于Java项目,Maven、Gradle或Ant等专业的构建工具是更常见的选择,它们提供了更强大的依赖管理、项目结构约定和生命周期管理功能,但在某些简单场景或特定需求下,Makefile仍然可以用来自动化Java程序的编译和执行。
一个基本的Makefile可以帮助我们定义编译规则和清理操作。以下是一个用于编译Java源代码的Makefile示例:
# 定义Java编译器和编译标志
JFLAGS = -g
JC = javac
# 定义后缀规则,将.java文件编译为.class文件
.SUFFIXES: .java .class
.java.class:
$(JC) $(JFLAGS) $*.java
# 定义需要编译的Java源文件
CLASSES = \
AVLTree.java
# 默认目标:编译所有类文件
default: classes
# classes目标:编译CLASSES中列出的所有Java文件
classes: $(CLASSES:.java=.class)
# clean目标:清理生成的.class文件
clean:
$(RM) *.classMakefile解释:
- JFLAGS 和 JC:定义了编译器的选项和命令。
- .SUFFIXES 和 .java.class:这是一个隐式规则,告诉make如何将.java文件转换为.class文件。$*.java表示不带扩展名的文件名。
- CLASSES:列出了所有需要编译的Java源文件。
- default: classes:当在命令行中只输入make时,会默认执行classes目标。
- classes: $(CLASSES:.java=.class):这个目标依赖于所有CLASSES中列出的Java文件对应的.class文件。make会根据.java.class规则自动编译这些文件。
- clean:用于删除所有生成的.class文件,保持工作目录整洁。
通过这个Makefile,运行make命令将编译AVLTree.java,生成AVLTree.class。运行make clean将删除生成的.class文件。
3. 整合命令行参数与程序执行
要让Makefile不仅编译程序,还能执行它并传递命令行参数,我们需要添加一个新的目标(例如run)到Makefile中,并将其设置为默认目标。
修改后的Makefile如下:
# 定义Java编译器和编译标志
JFLAGS = -g
JC = javac
# 定义Java虚拟机执行命令
JVM = java
# 定义类路径,这里是当前目录
CLASSPATH = .
# 定义后缀规则,将.java文件编译为.class文件
.SUFFIXES: .java .class
.java.class:
$(JC) $(JFLAGS) $*.java
# 定义需要编译的Java源文件
CLASSES = \
AVLTree.java
# 定义主类名(不带.java或.class扩展名)
MAIN_CLASS = AVLTree
# 定义命令行参数文件
INPUT_FILE = inputs.txt
# 将默认目标改为run,即执行程序
default: run
# classes目标:编译CLASSES中列出的所有Java文件
classes: $(CLASSES:.java=.class)
# run目标:编译后执行Java程序并传递命令行参数
run: classes
$(JVM) -cp $(CLASSPATH) $(MAIN_CLASS) $(INPUT_FILE)
# clean目标:清理生成的.class文件
clean:
$(RM) *.class修改点及解释:
- JVM 和 CLASSPATH 变量: 增加了JVM和CLASSPATH变量,使命令更具可读性和可维护性。
- MAIN_CLASS 变量: 定义了要执行的主类名,方便管理。
- INPUT_FILE 变量: 定义了作为命令行参数传递的文件名。
- default: run: 将默认目标从classes更改为run。这意味着当你只输入make时,它会先确保所有类都已编译(run目标依赖于classes),然后执行run目标。
-
run: classes:
- 这个目标依赖于classes目标,确保在执行之前所有Java文件都已编译。
- $(JVM) -cp $(CLASSPATH) $(MAIN_CLASS) $(INPUT_FILE):这是执行Java程序的命令。它使用java命令,指定类路径为当前目录,然后运行AVLTree类,并将inputs.txt作为命令行参数传递给它。
现在,在命令行中输入make,Makefile将首先编译AVLTree.java,然后执行java -cp . AvlTree inputs.txt命令。
4. 注意事项与最佳实践
构建工具选择: 对于任何规模的Java项目,强烈建议使用Maven或Gradle。它们提供了标准化的项目结构、强大的依赖管理、插件生态系统以及统一的构建生命周期,远比Makefile更适合Java开发。Makefile更适用于非常简单的脚本自动化或非Java项目的构建。
类路径 (Classpath): -cp .是指定当前目录为类路径的常用方式。如果你的.class文件位于子目录中,你需要相应地调整类路径,例如-cp bin或-cp ".:lib/*"(包含当前目录和lib目录下所有jar包)。
-
参数化: 在上述示例中,INPUT_FILE被硬编码在Makefile中。如果需要传递不同的文件,可以考虑使用Makefile的变量覆盖功能,例如:
make run INPUT_FILE=another_input.txt
在Makefile中,可以将INPUT_FILE定义为:
INPUT_FILE ?= inputs.txt # 如果未在命令行指定,则使用默认值
文件名与类名: 再次强调,java命令后面跟的是主类的名称(例如AvlTree),而不是文件名(AvlTree.java)或字节码文件名(AvlTree.class)。
错误处理与依赖管理: Makefile在处理复杂的Java依赖(如外部库JAR包)和错误处理方面远不如Maven/Gradle灵活和强大。当项目增长时,Makefile的维护成本会迅速增加。
总结
通过本文,我们了解了Java程序的标准编译和执行流程,并学习了如何利用Makefile来自动化这些步骤,尤其是在需要传递命令行参数时。尽管Makefile并非Java项目的首选构建工具,但理解其工作原理有助于更好地掌握构建自动化概念。在实际开发中,为了项目的可维护性和可扩展性,务必优先考虑使用Maven或Gradle等专业的Java构建工具。










