
本文详细阐述了Java程序的标准编译与执行流程,并探讨了如何利用Makefile来自动化这一过程,尤其是在需要传递命令行参数时。文章强调了Java项目通常更倾向于使用Maven或Gradle等专业构建工具,但仍提供了Makefile的实现示例,以帮助理解其基本原理和应用。
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 = -gJC = 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) *.class
Makefile解释:
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 = -gJC = 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构建工具。
以上就是Java程序与Makefile:编译、运行及命令行参数传递实践的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/70320.html
微信扫一扫
支付宝扫一扫