在没有IDE的情况下编译和运行Java

最近一个名为“ 不使用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 。 下一个屏幕快照显示了此结果。

javacCompilingPackageDirectoryButToSameDirForClasses

因为我们没有为.class文件指定目标目录,所以默认情况下将它们与编译它们的源.java文件放在同一目录中。 我们可以使用-d选项纠正这种情况。 我们的命令现在可以运行,例如,作为javac -sourcepath src -d classes src\dustin\examples\*.java 。 如前所述,指定的目标目录( classes )必须已经存在。 完成后,该命令会将.class文件放置在指定目录中,如下一个屏幕快照所示。

javacDoptionPlacesClassesCorrectDir

将Java源文件编译为指定目录中的相应.class文件后,我们现在可以使用Java应用程序启动器命令行工具java运行该应用程序。 只需按照java -help或java工具页面上显示的说明并使用-classpath (或-cp )选项指定.class文件的位置即可完成。 在下一个屏幕快照中演示了使用这两种方法指定classes目录是在其中查找.class文件的位置。 最后一个参数是具有要执行的main功能的类的标准名称(整个Java程序包)。 在下一个屏幕快照中演示的命令是java -cp classes dustin.examples.Mainjava -classpath classes dustin.examples.Main

javaRunningSimpleExamplesParentChildMain

使用Ant构建并运行

对于最简单的Java应用程序,使用javacjava分别构建和执行该应用程序非常简单,如刚刚演示的那样。 随着应用程序的参与度提高(例如,多个包/目录中存在的代码或第三方库和框架上更复杂的类路径依赖关系),这种方法可能变得笨拙。 Apache Ant是Java构建工具“三巨头 ”中最古老的一种,已在数千个应用程序和部署中使用。 正如我在上一篇博客文章中讨论的那样 ,很容易创建一个非常基本的Ant构建文件,尤其是如果该文件以我在该文章中概述的模板开头。

下一个代码清单是针对Ant build.xml文件的,该文件可用于将.java文件编译为.class文件,然后运行dustin.examples.Main类,就像上面对javacjava所做的一样。

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”),以使该示例尽可能简单,并使其与使用javacjava的上一个示例接近。 还要注意,我为javac Ant任务包括了将“ debug”设置为“ true”的原因,因为在Ant的默认设置中它不是true,而在javac的默认值中是true。 毫不奇怪,Ant的javac任务和java任务与命令工具javacjava非常相似。

因为我使用了Ant在未明确指定构建文件时会使用的默认名称( build.xml ),并且因为我为该构建文件提供了“运行”目标作为“默认”,并且因为我将“编译”作为依赖项而包括在内要运行“运行”目标,并且由于Ant在我环境的路径上,因此我需要在命令行上执行以使Ant编译并运行该示例的所有操作是在带有build.xml文件的目录中键入“ ant”。 下一个屏幕快照对此进行了演示。

antCompilingAndRunningSimpleJavaApp

尽管我演示了如何使用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文件,如下一个屏幕快照所示。

mavenCompilingJavaCode

我还可以使用命令mvn exec:java -Dexec.mainClass=dustin.examples.Main使用Maven运行编译后的应用程序,这将在下一个屏幕快照中演示。

mavenRunningSimpleApp

与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”,这将在下一个屏幕快照中演示。

gradleBuildAndRunSimpleExampleJavacJava

即使是最大的怀疑论者也不得不承认,对于这个简单的示例,Gradle的构建非常巧妙。 它结合了依赖某些约定和假设的简洁性,以及根据需要覆盖选择默认值的非常简单的机制。 它使用Groovy而不是XML的事实也非常吸引人!

与Ant和Maven一样,我倾向于仅使用这些工具进行构建,通常通常直接使用java或调用java的脚本来运行已编译的.class文件。 顺便说一句,我通常也将这些.class存档到运行中的JAR中,但这超出了本文的范围。

结论

IDE通常对于构建简单的应用程序和示例不是必需的,甚至可能比最简单的示例值得拥有更多的开销。 在这种情况下,直接应用javacjava来构建和运行示例相当容易。 随着示例变得越来越复杂,诸如Ant,Maven或Gradle之类的构建工具变得越来越有吸引力。 许多IDE支持这些构建工具的事实意味着,如果在简单应用程序发展成为一个成熟的项目时确定需要IDE支持,则开发人员可以使用在过程中较早创建的构建工具来过渡到IDE。

翻译自: https://www.javacodegeeks.com/2014/04/compiling-and-running-java-without-an-ide.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/363925.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java statement 返回类型,6.3 返回类型和返回语句 | Return type Return statement

无返回值函数对于返回类型是void的函数&#xff0c;return后不跟表达式。最后的 return; 可以没有而让程序隐式执行。在void函数中的return语句还有提前使函数退出的作用而不进行接下来的计算&#xff0c;如定义一个swap函数在二者相同时不继续运算直接退出&#xff1a;void sw…

异步调用 (转载)

异步操作通常用于执行完成时间可能较长的任务&#xff0c;如打开大文件、连接远程计算机或查询数据库。异步操作在主应用程序线程以外的线程中执行。应用程序调用方法异步执行某个操作时&#xff0c;应用程序可在异步方法执行其任务时继续执行。.NET框架能够对任何方法进行异步…

《Effective Java》读书笔记 Item 1:考虑静态工厂方法,而不是构造器

众所周知&#xff0c;要想能获取一个类的实例&#xff0c;该类得要提供一个public的构造器。但是《Effective Java》书中说还有一个方法&#xff0c;那就是提供静态工厂方法&#xff08;static factory method&#xff09;&#xff0c;该方法时静态&#xff0c;同时可以根据参数…

[转]MySQL 表锁和行锁机制

本文转自&#xff1a;http://www.cnblogs.com/itdragon/p/8194622.html MySQL 表锁和行锁机制 行锁变表锁&#xff0c;是福还是坑&#xff1f;如果你不清楚MySQL加锁的原理&#xff0c;你会被它整的很惨&#xff01;不知坑在何方&#xff1f;没事&#xff0c;我来给你们标记几个…

使用JS实现文字搬运工

使用JS实现文字搬运工 效果图&#xff1a; 代码如下&#xff0c;复制即可使用&#xff1a; <!DOCTYPE html> <html><head><meta http-equiv"Content-Type" content"text/html; charsetUTF-8"> <title>使用JS实现文字搬运工&…

ActiveMQ –经纪人网络解释–第3部分

现在&#xff0c;我们已经在本博客系列的第1部分和第2 部分中了解了ActiveMQ网络连接器的基础&#xff0c;在第3部分中&#xff0c;我们将研究ActiveMQ如何平衡连接到代理网络的使用者。 介绍 当可以无序处理队列中的消息时通常使用并发使用者&#xff0c;通常可以提高消息吞吐…

PJMEDIA之录音器的使用(capture sound to avi file)

为了熟悉pjmedia的相关函数以及使用方法&#xff0c;这里练习了官网上的一个录音器的例子。 核心函数&#xff1a; pj_status_t pjmedia_wav_writer_port_create(pj_pool_t * pool, const char * filename, unsigned clock_rate, unsigned channel_count, unsigned samples…

2007年暑期总结

又是一个没有回家的假期&#xff0c;一个月的鸡蛋炒拉面&#xff0c;一个月沉浸在编程的世界里……收获早已超越了技术本身&#xff0c;项目的实践&#xff0c;系统的架构&#xff0c;……更多的&#xff0c;是朋友们在一起的乐趣。A5闹鬼记&#xff0c;回去洗洗就睡了&#xf…

Linux下用户组、文件权限详解=------转载文

转载自-----原文地址&#xff1a; https://www.cnblogs.com/123-/p/4189072.html Linux下用户组、文件权限详解 用户组 在linux中的每个用户必须属于一个组&#xff0c;不能独立于组外。在linux中每个文件有所有者、所在组、其它组的概念 - 所有者 - 所在组 - 其它组 - 改变用…

风暴事件处理器–每个工作者的GC日志文件

在过去的三个月中&#xff0c;我正在与一个新团队合作&#xff0c;为电信领域的大数据分析构建产品。 Storm事件处理器是我们使用的主要框架之一&#xff0c;而且确实很棒。 您可以阅读其官方文档&#xff08;已改进&#xff09;中的更多详细信息。 Storm使用Workers来完成您…

JS实现逼真的雪花飘落特效

逼真的雪花飘落特效 效果图&#xff1a; 图片素材 &#xff1a; --> ParticleSmoke.png 代码如下&#xff0c;复制即可使用&#xff1a; <!doctype html> <html> <head> <meta charset"UTF-8"> <meta name"renderer" conte…

本地执行php查看内存占用,查看页面执行php占用内存情况

今天头脑一热&#xff0c;想看一下页面在执行的过程中占用了多少内存&#xff0c;我也不知道这样做的目的是什么&#xff0c;可能是出于我的惯性思维吧。不过这样做也不是完全没用&#xff0c;你可以清楚的知道哪些页面占用的内存比较多&#xff0c;特别是对于使用共用主机的网…

阴影及定位

阴影及定位 隐藏及阴影 标签隐藏 1、显示方式 display display: none; /*表示在页面中隐藏&#xff0c;并且不占位&#xff0c;但是重新显示出来又会占位*/ 2、透明度 opacity opacity: 0; /* 0 代表完全透明 1 代表完全显示 但是即使是透明了也会在页面中占位*/ /* 显示方式透…

ABAP--关于ABAP流程处理的一些命令的说明(stop,exit,return,check,reject)

Stop 命令使用该命令的程序位置INITIALIZATION, AT SELECTION-SCREEN, START-OF-SELECTION和GET 事件中处理说明1、 当在INITIALIZATION事件执行该命令&#xff0c;系统将直接触发应用服务器和客户端屏幕元素的发送&#xff1b;2、 在其他事件中将直接触发END-OF-SELECTION事件…

[Code+#3]寻找车位

[Code#3]寻找车位 挺厉害的线段树题 m<n&#xff0c;所以n<2000,并且只有1000次修改询问&#xff0c;mqlogn的复杂度可以接受&#xff01; 求全局&#xff1f; 对行(n)建一个线段树。 线段树中维护的东西&#xff0c;一定可以包含所有“完全包含在”这个横条中的最大正方…

ActiveMQ –经纪人网络解释–第2部分

在此博客中&#xff0c;我们将看到双工网络连接器如何工作。 在上一部分中&#xff0c;我们从broker-1和broker-2创建了一个网络连接器。 我们能够看到当代理2上有一个使用者使用队列“ foo.bar”时&#xff0c;代理1上的队列“ foo.bar”的消息如何转发到代理2上的队列“ foo…

JQ实现情人节表白程序

JQ实现情人节表白页面 效果图&#xff1a; 表白利页&#xff0c;你值得拥有哦&#xff01; 代码如下&#xff0c;复制即可使用&#xff1a; <!doctype html> <html> <head> <meta charset"utf-8"> <title>JQ实现情人节表白程序<…

php 内部异步执行顺序,event_loop中不同异步操作的执行顺序

关于js的单线程、怎么创建一个异步任务都是老生常谈的话题了&#xff0c;我们今天就总结一下js不同的异步操作到底执行顺序如何。首先我们要明白js两种任务类型&#xff0c;一个是macrotask(宏任务)&#xff0c;一个是 microtask(微任务)。一个宏任务就是一个事件循环&#xff…

OpenGPU.org域名已经被劫持

这个域名被指向了上海黄浦区某地。 此时此刻&#xff0c;我的心情无比激动&#xff0c;“这年头&#xff0c;你要是不被‘墙’了&#xff0c;都不好意思出门”。 This domain name had been redirected to Shanghai, PRC. At this moment, I’m really excited that it’s my h…

Java 8日期时间API教程:LocalDateTime

该博文是Java 8中引入的有关Date Time API的系列教程的一部分。在本博文中&#xff0c;我将介绍LocalDateTime类中可用的一些方法。 LocalDateTime是一个不可更改的线程安全对象&#xff0c;它表示ISO-8601日历系统中没有时区的日期时间&#xff0c;例如2014-03-30T02&#xf…