
在深入探讨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
重要提示: 直接运行java AvlTree.java inputs.txt虽然在某些JDK版本中可能奏效,但它实际上是JDK的一个便利功能,在后台隐式地进行了编译,并非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解释:
通过这个Makefile,运行make命令将编译AVLTree.java,生成AVLTree.class。运行make clean将删除生成的.class文件。
要让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修改点及解释:
现在,在命令行中输入make,Makefile将首先编译AVLTree.java,然后执行java -cp . AvlTree inputs.txt命令。
构建工具选择: 对于任何规模的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构建工具。
以上就是Java程序与Makefile:编译、运行及命令行参数传递实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号