最近一个名为“ 不使用IDE编译Java软件包 ”的Java subreddit线程提出了一个问题:“是否有一个命令将软件包内的一组Java文件编译到一个单独的文件夹中(以下简称为bin),以及如何我会去运行新的类文件吗?” 该帖子的作者kylolink解释说:“当我开始使用Java时,我依靠Eclipse为我完成所有编译,而只是担心编写代码。” 我已经看过很多次了,实际上,这就是促使我(现在已经4岁)的博客文章GPS系统和IDE的原因:有用还是有害? 我喜欢强大的现代Java IDE,它们每天使我的生活变得更轻松,但是了解如何在没有它们的情况下构建和运行简单的Java示例也有很多好处。 这篇文章专注于如何做到这一点。
在我的博客文章“ 通过简单测试学习Java”中 ,我写了一些我有时喜欢使用简单的文本编辑器和命令行工具编写,构建和运行简单应用程序的方式。 我现在有一个很好的主意,我最喜欢的Java IDE需要多少“开销”,并尽早做出决定,使用IDE获得的好处是否足以保证“开销”。 在大多数实际应用中,毫无疑问,IDE的“开销”是值得的。 但是,对于最简单的示例应用程序,并非总是如此。 本文的其余部分显示了在这些情况下如何在没有IDE的情况下构建和运行Java代码。
要构建和执行的Java代码
为了使本文的讨论更加具体,我将使用一些非常简单的Java类,这些Java类通过组合或继承相互关联,并且位于相同名称的包(而不是未命名的包 )中,称为dustin.examples
。 其中两个类没有main
函数,而第三个类Main.java
确实具有一个main
函数,以允许演示如何在没有IDE的情况下运行该类。 接下来显示这三个类的代码清单。
Parent.java
package dustin.examples;public class Parent
{@Overridepublic String toString(){return "I'm the Parent.";}
}
Child.java
package dustin.examples;public class Child extends Parent
{@Overridepublic String toString(){return "I'm the Child.";}
}
Main.java
package dustin.examples;import static java.lang.System.out;public class Main
{private final Parent parent = new Parent();private final Child child = new Child();public static void main(final String[] arguments){final Main instance = new Main();out.println(instance.parent);out.println(instance.child);}
}
下一个屏幕快照显示了具有这些类.java
源文件的目录结构。 屏幕快照显示了源文件在表示包名称的目录层次结构中(由于使用dustin.examples
因此使用了dustin.examples
dustin/examples
),并且反映包的目录层次结构在名为src
的子目录下。 我还创建了classes
子目录(当前为空)来放置已编译的.class
文件,因为javac
该目录不存在时不会创建该目录。
使用Javac进行构建并使用Java运行
无论使用哪种方法来构建Java代码(Ant,Maven,Gradle或IDE),它最终都归结为javac 。 通过运行javac -help
可以看到Oracle / Sun提供的javac 命令行工具的标准选项,而通过运行javac -help -X
可以查看其他扩展选项。 在Windows或Unix / Linux的 javac
的工具文档中可以找到有关如何应用这些选项的更多详细信息。
如javac文档所述, -sourcepath
选项可用于表示源文件所在的目录。 在上面的屏幕快照中显示的目录结构中,这意味着,假设我正在C:\java\examples\javacAndJava\
目录中运行javac
命令,则我的命令中需要包含以下内容: javac -sourcepath src src\dustin\examples\*.java
。 下一个屏幕快照显示了此结果。
因为我们没有为.class
文件指定目标目录,所以默认情况下将它们与编译它们的源.java
文件放在同一目录中。 我们可以使用-d
选项纠正这种情况。 我们的命令现在可以运行,例如,作为javac -sourcepath src -d classes src\dustin\examples\*.java
。 如前所述,指定的目标目录( classes
)必须已经存在。 完成后,该命令会将.class文件放置在指定目录中,如下一个屏幕快照所示。
将Java源文件编译为指定目录中的相应.class
文件后,我们现在可以使用Java应用程序启动器命令行工具java运行该应用程序。 只需按照java -help
或java工具页面上显示的说明并使用-classpath
(或-cp
)选项指定.class
文件的位置即可完成。 在下一个屏幕快照中演示了使用这两种方法指定classes
目录是在其中查找.class
文件的位置。 最后一个参数是具有要执行的main
功能的类的标准名称(整个Java程序包)。 在下一个屏幕快照中演示的命令是java -cp classes dustin.examples.Main
和java -classpath classes dustin.examples.Main
。
使用Ant构建并运行
对于最简单的Java应用程序,使用javac
和java
分别构建和执行该应用程序非常简单,如刚刚演示的那样。 随着应用程序的参与度提高(例如,多个包/目录中存在的代码或第三方库和框架上更复杂的类路径依赖关系),这种方法可能变得笨拙。 Apache Ant是Java构建工具“三巨头 ”中最古老的一种,已在数千个应用程序和部署中使用。 正如我在上一篇博客文章中讨论的那样 ,很容易创建一个非常基本的Ant构建文件,尤其是如果该文件以我在该文章中概述的模板开头。
下一个代码清单是针对Ant build.xml
文件的,该文件可用于将.java
文件编译为.class
文件,然后运行dustin.examples.Main
类,就像上面对javac
和java
所做的一样。
build.xml
<?xml version="1.0" encoding="UTF-8"?>
<project name="BuildingSansIDE" default="run" basedir="."><description>Building Simple Java Applications Without An IDE</description><target name="compile"description="Compile the Java code."><javac srcdir="src"destdir="classes"debug="true"includeantruntime="false" /></target><target name="run" depends="compile"description="Run the Java application."><java classname="dustin.examples.Main" fork="true"><classpath><pathelement path="classes"/></classpath></java></target>
</project>
我没有使用过Ant属性,也没有包括我通常包含的公共目标(例如“ clean”和“ javadoc”),以使该示例尽可能简单,并使其与使用javac
和java
的上一个示例接近。 还要注意,我为javac Ant任务包括了将“ debug”设置为“ true”的原因,因为在Ant的默认设置中它不是true,而在javac的默认值中是true。 毫不奇怪,Ant的javac任务和java任务与命令工具javac
和java
非常相似。
因为我使用了Ant在未明确指定构建文件时会使用的默认名称( build.xml
),并且因为我为该构建文件提供了“运行”目标作为“默认”,并且因为我将“编译”作为依赖项而包括在内要运行“运行”目标,并且由于Ant在我环境的路径上,因此我需要在命令行上执行以使Ant编译并运行该示例的所有操作是在带有build.xml
文件的目录中键入“ ant”。 下一个屏幕快照对此进行了演示。
尽管我演示了如何使用Ant编译和运行简单的Java应用程序,但是我通常仅使用Ant编译并使用java
(或者如果类路径令人讨厌的话,会调用java
的脚本)运行。
使用Maven构建和运行
尽管Ant是第一个主流的Java构建工具,但是Apache Maven最终获得了自己的显著成就,这在很大程度上要归功于它采用惯例进行配置并支持库的通用存储库。 当代码和生成的对象符合其标准目录布局时,Maven最容易使用。 不幸的是,我的示例没有遵循此目录结构,但是Maven允许我们覆盖期望的默认目录结构 。 下一个代码清单是针对Maven POM文件的,该文件将覆盖源目录和目标目录,并为使用Maven 3.2.1的Maven构建提供其他最低要求的元素。
pom.xml
<project><modelVersion>4.0.0</modelVersion><groupId>dustin.examples</groupId><artifactId>CompilingAndRunningWithoutIDE</artifactId><version>1</version><build><defaultGoal>compile</defaultGoal><sourceDirectory>src</sourceDirectory><outputDirectory>classes</outputDirectory><finalName>${project.artifactId}-${project.version}</finalName></build>
</project>
因为上面的pom.xml
文件将“ defaultGoal”指定为“ compile”,并且因为pom.xml
是Maven可执行文件(mvn)查找的默认自定义POM文件,并且因为Maven安装的bin
目录在我的路径上,所以我只需要运行“ mvn”来编译.class
文件,如下一个屏幕快照所示。
我还可以使用命令mvn exec:java -Dexec.mainClass=dustin.examples.Main
使用Maven运行编译后的应用程序,这将在下一个屏幕快照中演示。
与Ant一样,我通常不使用Maven运行我的简单Java应用程序,而是在编译后的代码上使用java
(或使用直接为长类路径调用java
的脚本)。
使用Gradle构建和运行
Gradle是三种主要Java构建工具中最年轻,最时髦和最时髦的。 有时,我对某种时髦的内容持怀疑态度,但是我发现Gradle有很多令人喜欢的地方 (用Groovy而不是XML编写,内置的Ant支持,内置的Ivy支持,按约定容易被覆盖的配置) ,Maven存储库支持等)。 下一个示例显示了一个Gradle构建文件,该文件可用于编译和运行简单应用程序,该应用程序是本文的主要示例代码。 它改编自我在博客文章Simple Gradle Java Plugin Customization中介绍的示例。
build.gradle
apply plugin: 'java'
apply plugin: 'application'// Redefine where Gradle should expect Java source files (*.java)
sourceSets {main {java {srcDirs 'src'}}
}// Redefine where .class files are written
sourceSets.main.output.classesDir = file("classes")// Specify main class to be executed
mainClassName = "dustin.examples.Main"defaultTasks 'compileJava', 'run'
build.gradle
文件的前两行指定Java插件和Application插件的应用程序 ,从而自动为该版本带来了很多功能。 “ sourceSets”和“ sourceSets.main.output.classesDir”的定义允许分别覆盖Gradle的Java插件的Java源代码和已编译二进制类的默认目录。 “ mainClassName”允许显式指定应作为Application插件的一部分运行的类。 “ defaultTasks”行通过在命令行中简单地键入“ gradle”来指定要运行的任务:“ compileJava”是Java插件提供的标准任务,“ run”是Application插件提供的标准任务。 因为我将构建文件命名为build.gradle
并且因为我指定了默认任务'compileJava'和'run',并且因为我的路径上有Gradle安装bin
目录,所以构建和运行示例所需要做的就是键入“ gradle”,这将在下一个屏幕快照中演示。
即使是最大的怀疑论者也不得不承认,对于这个简单的示例,Gradle的构建非常巧妙。 它结合了依赖某些约定和假设的简洁性,以及根据需要覆盖选择默认值的非常简单的机制。 它使用Groovy而不是XML的事实也非常吸引人!
与Ant和Maven一样,我倾向于仅使用这些工具进行构建,通常通常直接使用java
或调用java
的脚本来运行已编译的.class
文件。 顺便说一句,我通常也将这些.class
存档到运行中的JAR中,但这超出了本文的范围。
结论
IDE通常对于构建简单的应用程序和示例不是必需的,甚至可能比最简单的示例值得拥有更多的开销。 在这种情况下,直接应用javac
和java
来构建和运行示例相当容易。 随着示例变得越来越复杂,诸如Ant,Maven或Gradle之类的构建工具变得越来越有吸引力。 许多IDE支持这些构建工具的事实意味着,如果在简单应用程序发展成为一个成熟的项目时确定需要IDE支持,则开发人员可以使用在过程中较早创建的构建工具来过渡到IDE。
翻译自: https://www.javacodegeeks.com/2014/04/compiling-and-running-java-without-an-ide.html