在没有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,一经查实,立即删除!

相关文章

[转]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;通常可以提高消息吞吐…

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

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

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

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

阴影及定位

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

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实现情人节表白程序<…

消息队列使用的四种场景介绍(转)

消息队列中间件是分布式系统中重要的组件&#xff0c;主要解决应用耦合&#xff0c;异步消息&#xff0c;流量削锋等问题 实现高性能&#xff0c;高可用&#xff0c;可伸缩和最终一致性架构 使用较多的消息队列有ActiveMQ&#xff0c;RabbitMQ&#xff0c;ZeroMQ&#xff0c;Ka…

一步一步SharePoint 2007之三十一:实现文档Event Handler(3)——附加Handler程序

摘要  本篇文章将介绍实现文档Event Handler的第三部分——附加Handler程序。正文  下面将记录每一步的操作过程。  1、首先打开我的网站&#xff0c;依次点击Document Center、Documents&#xff0c;进入Documents列表页面。  2、在Documents列表界面中点击Settings&a…

Day2 HTML基本标签元素

Day2 HTML基本标签元素 HTML: 超文本标记语言(HyperText Mark-up Language ) 1.作用&#xff1a;写网页结构    2.HTML不区分大小写&#xff0c;建议小写   3.文件后缀 .html 或者 .htm   4.html由浏览器解析执行. 由上往下&#xff0c;由左往右 1) HTML标…

作用域、执行环境、闭包(四)

本文也同步发表在我的公众号“我的天空” 上一期我们已经介绍了闭包&#xff0c;由于闭包可以延长函数内部的变量的生存周期&#xff0c;因此我们可以将不需要暴露在全局的变量封装成函数的内部变量&#xff0c;从而避免代码污染。 譬如要实现一个简单的累加器&#xff0c;为了…

php static_castunsigned int,C++ static_cast、dynamic_cast、const_cast和reinterpret_cast(四种类型转换运算符)...

上节讲到&#xff0c;隐式类型转换是安全的&#xff0c;显式类型转换是有风险的&#xff0c;C语言之所以增加强制类型转换的语法&#xff0c;就是为了强调风险&#xff0c;让程序员意识到自己在做什么。但是&#xff0c;这种强调风险的方式还是比较粗放&#xff0c;粒度比较大&…

AngularJS(九):路由

本文也同步发表在我的公众号“我的天空” AngularJS路由 AngularJS路由可以让我们通过不同的URL访问不同页面&#xff08;似乎是废话&#xff09;&#xff0c;其价值主要体现在单页面的web应用中&#xff08;single page web application&#xff0c;SPA&#xff09;&#xff0…

[C3W2] Structuring Machine Learning Projects - ML Strategy 2

第二周&#xff1a;机器学习策略&#xff08;2&#xff09;&#xff08;ML Strategy&#xff08;2&#xff09;&#xff09; 误差分析&#xff08;Carrying out error analysis&#xff09; 你好&#xff0c;欢迎回来&#xff0c;如果你希望让学习算法能够胜任人类能做的任务&a…

mysql语句执行顺序图示

转载于:https://www.cnblogs.com/whalesea/p/10382227.html

AngularJS(三):重复HTML元素、数据绑定

本文也同步发表在我的公众号“我的天空” 重复HTML元素 在前端的页面编写中&#xff0c;我们会经常遇到重复HTML元素&#xff0c;譬如绘制表格、菜单等&#xff0c;如以下代码显示一个简单的li列表&#xff1a; <body> <ul id"ul_cities"> </ul…

使用表中的数组数据类型

在这篇文章中&#xff0c;我想跟进我以前关于Oracle集合数据类型的文章 &#xff0c;并且我将集中精力使用af&#xff1a;table组件中的oracle.jbo.domain.Array属性。 因此&#xff0c;在我的数据库中&#xff0c;我具有以下SQL类型&#xff1a; create or replace type var…

linux 文件inode,linux文件系统-inode学习整理

linux文件系统-inode学习整理介绍linux文件系统可讲的模块有很多&#xff0c;包括文件系统整体架构、文件系统分类、虚拟文件系统以及文件系统存储结构等等&#xff0c;本文主要介绍的是文件系统的存储结构&#xff0c;也就是本文的重点-inode。文件存储结构首先从开天辟地开始…

pygame-KidsCanCode系列jumpy-part8-记录历史最高分

通常在多玩家的游戏中&#xff0c;每个玩家都会有自己的得分&#xff0c;最高分数会成为该游戏的最佳记录。这一篇&#xff0c;学习下如何记录最高得分&#xff1a;&#xff08;为了简化代码&#xff0c;本文采用文件方式&#xff0c;仅记录本机得分&#xff0c;明白原理后&…